EJB-지속성

EJB 2.0에서 사용되는 엔티티 빈인 EJB 3.0은 대부분 지속성 메커니즘으로 대체됩니다. 이제 엔티티 빈은 테이블과 매핑되는 간단한 POJO입니다.

다음은 지속성 API의 주요 행위자입니다-

  • Entity− 데이터 저장소 레코드를 나타내는 영구 객체. 직렬화하는 것이 좋습니다.

  • EntityManager− 영구 객체 (엔티티)에 대한 추가 / 삭제 / 업데이트 / 찾기와 같은 데이터 작업을 수행하기위한 지속성 인터페이스. 또한 다음을 사용하여 쿼리를 실행하는 데 도움이됩니다.Query 상호 작용.

  • Persistence unit (persistence.xml) − 지속성 단위는 지속성 메커니즘의 속성을 설명합니다.

  • Data Source (*ds.xml)− 데이터 소스는 연결 URL과 같은 데이터 저장소 관련 속성을 설명합니다. 사용자 이름, 비밀번호 등

EJB 지속성 메커니즘을 시연하려면 다음 작업을 수행해야합니다.

  • Step 1 − 데이터베이스에 테이블을 생성합니다.

  • Step 2 − 테이블에 해당하는 Entity 클래스를 생성합니다.

  • Step 3 − 데이터 소스 및 지속성 단위를 생성합니다.

  • Step 4 − EntityManager 인스턴스가있는 상태 비 저장 EJB를 생성합니다.

  • Step 5− 상태 비 저장 EJB를 업데이트합니다. 레코드를 추가하고 엔티티 관리자를 통해 데이터베이스에서 레코드를 가져 오는 메서드를 추가합니다.

  • Step 6 − 콘솔 기반 애플리케이션 클라이언트는 데이터베이스에 데이터를 유지하기 위해 상태 비 저장 EJB에 액세스합니다.

테이블 생성

테이블 만들기 books 기본 데이터베이스 postgres.

CREATE TABLE books (
   id     integer PRIMARY KEY,
   name   varchar(50)
);

엔티티 클래스 생성

//mark it entity using Entity annotation 
//map table name using Table annotation
@Entity
@Table(name="books")
public class Book implements Serializable{
    
   private int id;
   private String name;

   public Book() {        
   }

   //mark id as primary key with autogenerated value
   //map database column id with id field
   @Id
   @GeneratedValue(strategy= GenerationType.IDENTITY)
   @Column(name="id")
   public int getId() {
      return id;
   }
   ...
}

데이터 소스 및 지속성 단위 만들기

DataSource (jboss-ds.xml)

<?xml version = "1.0" encoding = "UTF-8"?>
<datasources>
   <local-tx-datasource>
      <jndi-name>PostgresDS</jndi-name>
      <connection-url>jdbc:postgresql://localhost:5432/postgres</connection-url>
      <driver-class>org.postgresql.driver</driver-class>
      <user-name>sa</user-name>
      <password>sa</password>
      <min-pool-size>5</min-pool-size>
      <max-pool-size>20</max-pool-size>
      <idle-timeout-minutes>5</idle-timeout-minutes>
   </local-tx-datasource>
</datasources>

지속성 단위 (persistence.xml)

<persistence version = "1.0" xmlns = "http://java.sun.com/xml/ns/persistence" xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_1_0.xsd">

   <persistence-unit name = "EjbComponentPU" transaction-type = "JTA">
      <jta-data-source>java:/PostgresDS</jta-data-source>
      <exclude-unlisted-classes>false</exclude-unlisted-classes>
      <properties/>
   </persistence-unit>
   
   <persistence-unit name = "EjbComponentPU2" transaction-type = "JTA">
      <provider>org.hibernate.ejb.HibernatePersistence</provider>
      <jta-data-source>java:/PostgresDS</jta-data-source>
      <exclude-unlisted-classes>false</exclude-unlisted-classes>
	  
      <properties>
         <property name="hibernate.hbm2ddl.auto" value="update"/>
      </properties>
   </persistence-unit>
   
</persistence>

EntityManager 인스턴스가있는 상태 비 저장 EJB 만들기

@Stateless
public class LibraryPersistentBean implements LibraryPersistentBeanRemote {
	
   //pass persistence unit to entityManager.
   @PersistenceContext(unitName="EjbComponentPU")
   private EntityManager entityManager;         

   public void addBook(Book book) {
      entityManager.persist(book);
   }    

   public List<Book> getBooks() {        
      return entityManager.createQuery("From Books").getResultList();
   }
   ...
}

EJB 모듈을 빌드 한 후에는 다음 섹션에서 생성 할 stateless bean에 액세스 할 클라이언트가 필요합니다.

예제 애플리케이션

EJB 지속성 메커니즘을 테스트하기위한 테스트 EJB 애플리케이션을 만들어 보겠습니다.

단계 기술
1

EJB- 애플리케이션 만들기 장에 설명 된대로 com.tutorialspoint.entity 패키지 아래에 EjbComponent 라는 이름으로 프로젝트를 만듭니다 . 이 장에서 EJB 지속성 개념을 이해하기 위해 EJB- 응용 프로그램 만들기 장 에서 만든 프로젝트를 사용할 수도 있습니다 .

2

만들기 Book.java을 패키지에서 com.tutorialspoint.entity 아래와 같이 수정합니다.

만들기 LibraryPersistentBean.javaLibraryPersistentBeanRemote을 에 설명 된대로 EJB - 응용 프로그램 생성 장을 아래와 같이 수정할.

4

만들기 의 jboss-ds.xml 의를EjbComponent > setup 폴더 의 persistence.xmlEjbComponent > src > conf 폴더. 이 폴더는 Netbeans의 파일 탭에서 볼 수 있습니다. 위에 표시된대로 이러한 파일을 수정하십시오.

5

응용 프로그램을 정리하고 빌드하여 비즈니스 논리가 요구 사항에 따라 작동하는지 확인합니다.

6

마지막으로 JBoss Application Server에 jar 파일 형식으로 애플리케이션을 배포합니다. JBoss Application Server는 아직 시작되지 않은 경우 자동으로 시작됩니다.

7

이제 EJB-응용 프로그램 만들기 장에서 설명한 것과 동일한 방식으로 콘솔 기반 응용 프로그램 인 EJB 클라이언트를 만듭니다.Create Client to access EJB. 아래와 같이 수정합니다.

EJBComponent (EJB 모듈)

Book.java

package com.tutorialspoint.entity;

import java.io.Serializable;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.EntityListeners;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;

@Entity
@Table(name="books")
public class Book implements Serializable{
    
   private int id;
   private String name;

   public Book() {        
   }

   @Id
   @GeneratedValue(strategy= GenerationType.IDENTITY)
   @Column(name="id")
   public int getId() {
      return id;
   }

   public void setId(int id) {
      this.id = id;
   }

   public String getName() {
      return name;
   }

   public void setName(String name) {
      this.name = name;
   }    
}

LibraryPersistentBeanRemote.java

package com.tutorialspoint.stateless;

import com.tutorialspoint.entity.Book;
import java.util.List;
import javax.ejb.Remote;

@Remote
public interface LibraryPersistentBeanRemote {

   void addBook(Book bookName);

   List<Book> getBooks();
    
}

LibraryPersistentBean.java

package com.tutorialspoint.stateless;

import com.tutorialspoint.entity.Book;
import java.util.List;
import javax.ejb.Stateless;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;

@Stateless
public class LibraryPersistentBean implements LibraryPersistentBeanRemote {
    
   public LibraryPersistentBean() {
   }

   @PersistenceContext(unitName="EjbComponentPU")
   private EntityManager entityManager;         

   public void addBook(Book book) {
      entityManager.persist(book);
   }    

   public List<Book> getBooks() {
      return entityManager.createQuery("From Book").getResultList();
   }
}
  • JBOSS에 EjbComponent 프로젝트를 배포하자마자 jboss 로그를 확인하십시오.

  • JBoss는 세션 빈에 대한 JNDI 항목을 자동으로 생성했습니다. LibraryPersistentBean/remote.

  • 이 조회 문자열을 사용하여 유형의 원격 비즈니스 오브젝트를 가져옵니다. com.tutorialspoint.stateless.LibraryPersistentBeanRemote

JBoss 애플리케이션 서버 로그 출력

...
16:30:01,401 INFO  [JndiSessionRegistrarBase] Binding the following Entries in Global JNDI:
   LibraryPersistentBean/remote - EJB3.x Default Remote Business Interface
   LibraryPersistentBean/remote-com.tutorialspoint.stateless.LibraryPersistentBeanRemote - EJB3.x Remote Business Interface
16:30:02,723 INFO  [SessionSpecContainer] Starting jboss.j2ee:jar=EjbComponent.jar,name=LibraryPersistentBeanRemote,service=EJB3
16:30:02,723 INFO  [EJBContainer] STARTED EJB: com.tutorialspoint.stateless.LibraryPersistentBeanRemote ejbName: LibraryPersistentBean
16:30:02,731 INFO  [JndiSessionRegistrarBase] Binding the following Entries in Global JNDI:

   LibraryPersistentBean/remote - EJB3.x Default Remote Business Interface
   LibraryPersistentBean/remote-com.tutorialspoint.stateless.LibraryPersistentBeanRemote - EJB3.x Remote Business Interface
...

EJBTester (EJB 클라이언트)

jndi.properties

java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory
java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces
java.naming.provider.url=localhost
  • 이러한 속성은 Java 이름 지정 서비스의 InitialContext 개체를 초기화하는 데 사용됩니다.

  • InitialContext 객체는 상태 비 저장 세션 빈을 조회하는 데 사용됩니다.

EJBTester.java

package com.tutorialspoint.test;
   
import com.tutorialspoint.stateless.LibraryPersistentBeanRemote;
import java.io.BufferedReader;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.List;
import java.util.Properties;
import javax.naming.InitialContext;
import javax.naming.NamingException;

public class EJBTester {

   BufferedReader brConsoleReader = null; 
   Properties props;
   InitialContext ctx;
   {
      props = new Properties();
      try {
         props.load(new FileInputStream("jndi.properties"));
      } catch (IOException ex) {
         ex.printStackTrace();
      }
      try {
         ctx = new InitialContext(props);            
      } catch (NamingException ex) {
         ex.printStackTrace();
      }
      brConsoleReader = 
      new BufferedReader(new InputStreamReader(System.in));
   }
   
   public static void main(String[] args) {

      EJBTester ejbTester = new EJBTester();

      ejbTester.testEntityEjb();
   }
   
   private void showGUI() {
      System.out.println("**********************");
      System.out.println("Welcome to Book Store");
      System.out.println("**********************");
      System.out.print("Options \n1. Add Book\n2. Exit \nEnter Choice: ");
   }
   
   private void testEntityEjb() {

      try {
         int choice = 1; 

         LibraryPersistentBeanRemote libraryBean =
         LibraryPersistentBeanRemote)ctx.lookup("LibraryPersistentBean/remote");

         while (choice != 2) {
            String bookName;
            showGUI();
            String strChoice = brConsoleReader.readLine();
            choice = Integer.parseInt(strChoice);
            if (choice == 1) {
               System.out.print("Enter book name: ");
               bookName = brConsoleReader.readLine();
               Book book = new Book();
               book.setName(bookName);
               libraryBean.addBook(book);          
            } else if (choice == 2) {
               break;
            }
         }

         List<Book> booksList = libraryBean.getBooks();

         System.out.println("Book(s) entered so far: " + booksList.size());
         int i = 0;
         for (Book book:booksList) {
            System.out.println((i+1)+". " + book.getName());
            i++;
         }           
      } catch (Exception e) {
         System.out.println(e.getMessage());
         e.printStackTrace();
      }finally {
         try {
            if(brConsoleReader !=null) {
               brConsoleReader.close();
            }
         } catch (IOException ex) {
            System.out.println(ex.getMessage());
         }
      }
   }
}

EJBTester는 다음 작업을 수행합니다.

  • jndi.properties에서 속성을로드하고 InitialContext 객체를 초기화합니다.

  • testStatefulEjb () 메소드에서 jndi 조회는 "LibraryStatefulSessionBean / remote"라는 이름으로 수행되어 원격 비즈니스 객체 (상태 저장 ejb)를 얻습니다.

  • 그런 다음 사용자에게 라이브러리 저장소 사용자 인터페이스가 표시되고 선택 항목을 입력하라는 메시지가 표시됩니다.

  • 사용자가 1을 입력하면 시스템은 책 이름을 요청하고 stateless 세션 빈 addBook () 메소드를 사용하여 책을 저장합니다. Session Bean은 EntityManager 호출을 통해 데이터베이스에서 책을 유지합니다.

  • 사용자가 2를 입력하면 시스템은 Stateful 세션 빈 getBooks () 메소드를 사용하여 책을 검색하고 종료합니다.

  • 그런 다음 "LibraryStatelessSessionBean / remote"라는 이름으로 또 다른 jndi 검색이 수행되어 원격 비즈니스 객체 (상태 비 저장 EJB)를 다시 얻고 도서 목록이 완료됩니다.

클라이언트를 실행하여 EJB에 액세스

프로젝트 탐색기에서 EJBTester.java를 찾습니다. EJBTester 클래스를 마우스 오른쪽 버튼으로 클릭하고run file.

Netbeans 콘솔에서 다음 출력을 확인하십시오.

run:
**********************
Welcome to Book Store
**********************
Options 
1. Add Book
2. Exit 
Enter Choice: 1
Enter book name: Learn Java
**********************
Welcome to Book Store
**********************
Options 
1. Add Book
2. Exit 
Enter Choice: 2
Book(s) entered so far: 1
1. learn java
BUILD SUCCESSFUL (total time: 15 seconds)

클라이언트를 다시 실행하여 EJB에 액세스

EJB에 액세스하기 전에 JBoss를 다시 시작하십시오.

프로젝트 탐색기에서 EJBTester.java를 찾습니다. EJBTester 클래스를 마우스 오른쪽 버튼으로 클릭하고run file.

Netbeans 콘솔에서 다음 출력을 확인하십시오.

run:
**********************
Welcome to Book Store
**********************
Options 
1. Add Book
2. Exit 
Enter Choice: 1
Enter book name: Learn Spring
**********************
Welcome to Book Store
**********************
Options 
1. Add Book
2. Exit 
Enter Choice: 2
Book(s) entered so far: 2
1. learn java
2. Learn Spring
BUILD SUCCESSFUL (total time: 15 seconds)

위에 표시된 출력은 책이 영구 저장소에 저장되고 데이터베이스에서 검색됨을 나타냅니다.

이전 페이지