EJB中EntityManager的管理方式有两种:Container-managed EntityManager和Application-managed EntityManager
即容器管理的EntityManager和应用管理的EntityManager
在EJB中,EntityManager所进行的持久化的方式与Hibernate的方式是不同的。
1.在Hibernate的同一个事务中,通过getCurrentSession获取的session对象均为同一个,保存于threadlocal中,以保证对数据的操作为同一个对象。
2.而EJB中,以容器管理的EntityManager为例,通过注解注入后,在同一个事物中,在各个Bean中获取的EntityManager为不同的对象,但其背后所指向的 persistenceContext 为同一个,所以保证对一个对象进行操作。
1)容器管理的EntityManager
简单的说,就是使用注解,在程序启动的时候由容器自动注入的方式,这是一种普遍采用的方式。
@PersistenceContext(unitName="jbossDB") private EntityManager em;
在使用结束时,不需要自己关闭,有容器来管理。unitName为数据库资源的名字,有presidence.xml中定义
在该方式下,对应两种persistence类型:
1.transaction-scope persistence:由容器管理的persistence,其生命周期为一个transaction,这是默认的模式
一个transaction可认为是客户端向服务器端的一个请求,从开始到结束的一个周期
2.Extended-scope persistence:用于stateful session bean中,其生命周期随stateful的生命周期
@PersistenceContext(unitName="jbossDB",type=PersistenceContextType.EXTENDED) private EntityManager em;
其设置方式为通过PersistenceContextType来设置。
Stateful Session Bean 和 Stateless Session Bean最大的区别就是前者会保存客户端的信息,使得同一个客户端在一个Bean中多次访问的时候均为访问同一个Bean。
而后者则可看作单例模式,他不会保存客户端的信息,所以第二个客户端依然可以使用该Bean对象。
实体:
import java.io.Serializable; import javax.persistence.Entity; import javax.persistence.GeneratedValue; import javax.persistence.Id; import javax.persistence.Version; @Entity public class People implements Serializable{ /** * */ private static final long serialVersionUID = 20151228215045281L; @Id @GeneratedValue private int id; private String name; private String password; @Version private int version; 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; } public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } public int getVersion() { return version; } public void setVersion(int version) { this.version = version; } }
Bean继承的接口:
import com.welv.jpa.People; public interface PeopleManager { public void addPeople(People p); public People getPeople(int id); public void setName(String name); public void setPassword(String password); }
Bean的实现:
import javax.ejb.Remote; import javax.ejb.Stateful; import javax.persistence.EntityManager; import javax.persistence.EntityManagerFactory; import javax.persistence.PersistenceContext; import javax.persistence.PersistenceContextType; import javax.persistence.PersistenceUnit; import com.welv.ejb.PeopleManager; import com.welv.jpa.People; @Stateful @Remote public class PeopleManagerBean implements PeopleManager { @PersistenceContext(unitName="jbossDB",type=PersistenceContextType.EXTENDED) private EntityManager em; private People p; @Override public void addPeople(People p) { em.persist(p); } @Override public People getPeople(int id) { p = em.find(People.class, id); return p; } @Override public void setName(String name) { p.setName(name); } @Override public void setPassword(String password) { p.setPassword(password); } }
客户端代码:
import java.util.Properties; import javax.naming.Context; import javax.naming.InitialContext; import com.welv.ejb.PeopleManager; import com.welv.jpa.People; public class JpaEJBClient { public static void main(String[] arg) { try { final Properties jndiProperties = new Properties(); jndiProperties.put(Context.URL_PKG_PREFIXES, "org.jboss.ejb.client.naming"); final Context context = new InitialContext(jndiProperties); final String appName = ""; // 部署的jar文件的名字 final String moduleName = "EJB_06"; final String distinctName = ""; // 实现类的类名 final String beanName = "PeopleManagerBean"; // 接口类的全名 final String viewClassName = PeopleManager.class.getName(); String lookupStr = "ejb:" + appName + "/" + moduleName + "/" + distinctName + "/" + beanName + "!" + viewClassName+"?stateful"; System.out.println(lookupStr); PeopleManager ejb = (PeopleManager) context.lookup(lookupStr); // People p = new People(); // // p.setName("ll"); // p.setPassword("123456"); // // ejb.addPeople(p); People p = ejb.getPeople(1); ejb.setName("cccc"); ejb.setPassword("1234455"); } catch (Exception e) { System.out.println("============================================"); e.printStackTrace(); System.out.println("============================================"); System.out.println("请检查调用方法是否为远程"); } } }
从Bean的实现中可以看出,对People对象操作的一些方法是没有加载People对象的。如果是通过Stateless将接口暴露出去,这无法实现相应方法的功能。
a.由于使用的是StatefulBean,所以获取的Bean对象是有状态的,Stateful周期中多次的请求都是用同一个Bean,在远程使用getPeople()方法的时候,Session Bean中会
将People对象保存在this.People对象中,所以在同一个Stateful周期中使用其他的方法对People对象进行操作时就不需要重新load People对象。
b.若使用的是StatelessBean,则必须要在方法中重新加载,因为Stateless是不保存客户端信息的,也就是在同一个使用一个Stateless进行请求的时候,请求的Bean对象不
是同一个,所以如果不加载对象,会有NullPointException产生。
2)应用管理的EntityManager
由PersistenceUnit注入EntityManagerFactory中
@PersistenceUnit(unitName="jbossDB") private EntityManagerFactory emf;
在transaction结束时需要手动将获取的EntityManager关闭(closs);
**private static final long serialVersionUID = 20151228215045281L;作用:
该代码的作用用于匹配,在一般情况下,若EJB中的实体对象和客户端中的对象如果一致,则可以互相传递,但如果EJB中多加了属性,则再次访问的时候则会失败,而如果两边的代码中有相同的serialVersionUID则会认为对象相同,可以交互。
***由于刚了解该技术点,没有理解透