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.java 및 LibraryPersistentBeanRemote을 에 설명 된대로 EJB - 응용 프로그램 생성 장을 아래와 같이 수정할. |
4 | 만들기 의 jboss-ds.xml 의를EjbComponent > setup 폴더 의 persistence.xml 에EjbComponent > 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)
위에 표시된 출력은 책이 영구 저장소에 저장되고 데이터베이스에서 검색됨을 나타냅니다.