作者:狂放不羁
网址:http://yuquan-nana.iteye.com
>>>转载请注明出处!<<<
接着上篇“J2EE事务并发控制策略总结”,今天我接着写一下持久化上下文的传播问题。
目前持久层框架都有一个持久化上下文的概念,下面以比较流行的hibernate以及JPA来做一总结。
如果我们采用OO的方式开发系统,那么势必为了减低耦合,增加内聚,我们会通过细粒度的类来实现业务功能,那么这样就产生了一个问题,如何将持久化上下文在不同的类(这里面其实就是Dao类或者DDD里面的repository)中传播,比如传统的开发方式中,一个service里通过不同的Dao来访问数据库,那么怎么保证不同的Dao类中用的session以及以及与session对应的持久化上下文是同一个呢?这就涉及到了持久化上下文如何传播的问题。下面以我熟悉的hibernate以及JPA来来说明一下,在hibernate中,根据采用的底层事务的不同,需要采用不同的策略来实现:
1 Hibernate中持久化上下文的传播
1.1 采用jdbc事务
此时hibernate通过Threadlocal将当前的session与当前的线程绑定在一起,这样以来只要是同一个线程中的调用,那么获得的session都是同一个,具体来说就是配置hibernate.current_session_context_class属性为thread,这样以来hibernate内部就会通过CurrentSessionContext接口的实现类ThreadlocalSessionContext通过threadlocal来将session和当前线程绑定在一起。当调用SessionFactory的getCurrentSession()方法,返回的就是与当前线程绑定的session,从而解决了持久层上下文传播的问题。
1.2 采用JTA事务
此时hibernate内部是将当前的session以及对应的持久化上下文绑定到了全局的JTA事务上,这样以来我们通过sessionFactory的getCurrentSession()方法获得的就是与当前的JTA事务绑定的session.具体一点就是配置属性hibernate.current_session_context_class为jta,这样以来hibernate内部就是通过CurrentSessionContext接口的实现类JTASessionContext来将session与当前的JTA全局事务绑定在一起,因此当我们通过sf.getCurrentSessionContext()来获取session时,获得的就是与当前的JTA绑定到一起的session.
但是在此种情况下,需要特别注意一个问题:不能同时使用hibernate的Transaction接口与getCurrentSession(),因为当前的session是绑定到全局JTA事务中的,如果通过session.beginTransaction()来开始事务,这说明以前没有事务,既然没有事务存在,我们的session又是怎么绑定到全局JTA事务上的呢?所以一定要注意:当使用JTA事务,并且用了getCurrentSession()的方法时,一定不要用hiernate的native transaction 接口。但是我们如果我们用openSession(),那么就可以通过通过hibernate的native Transaction接口来控制JTA事务。
最后还需要弄清楚一点,Hibernate中还有一种绑定持久化上下文的方法,那就是通过设置hibernate.current_session_context_class属性为managed,hiberante内部就是通过CurrentSessionContext接口的实现类ManagedSessionContext来绑定的。在此种情况下主要是为了实现会话,会话在hibernate中的实现在下面介绍。
2 JPA中持久化上下文的传播
JPA中持久化上下文的传播根据采用不同的事务模型而不同,下面分别来说明:
2.1 采用resource-local事务模型
如果采用resource-local事务模型,此种情况也就是在非J2EE应用服务器的支持下使用。那么我们的持久化上下文的生命周期是与当前的EntityManager绑定到一起的,所以我们可以在不同的类中传播相同的EntityManager实例来达到传播事务上下文的传播。
2.2 采用JTA事务模型
在此种模型下面,我们有EJB容器的支持,持久化上下文的传播是借助于事务上下文来传播的,在说明如何传播前首先要明确EJB组件中两种不同的事务上下文的生命周期:
对于stateless session bean,持久化上下文的生命周期是与当前的系统事务一致的,这也就是无状态会话bean的事务型的持久化上下文,每当事务结束,持久化上下文也就结束了,所有持久化对象也就变为了脱管的(detached).
对于statefull session bean,持久化上下文的生命周期是与当前的有状态会话bean一致的,只有当有状态的会话bean从系统中移除的时候,持久化上下文才关闭,这也就是有状态会话bean的扩展的事务上下文。
搞清楚了这两种不同的事务上下文的生命周期以后,我们来说一下持久化上下文如何传播的问题。持久化上下文是通过当前系统事务来传播的,当一个EJB组件调用另一个EJB组件的时候,如果两个EJB组件的事务范围是一样的,那么持久化上下文就会传播,下面分几种情况来说明:
无状态会话bean之间调用:此时如果两个无状态会话bean在同一个事务中调用,那么持久化上下文就是同一个,通过当前事务来传播。
有状态会话bean调用无状态会话bean:此种情况下如果两者在同一个事务中,那么有状态会话bean的扩展的事务上下文会传播到无状态会话bean里,其实还是通过事务来传播。但是如果被调用的无状态会话bean不支持事务的话(事务属性设置为not support 或者never),那么此时持久化上下文不能传播(JPA规范规定扩展的持久化上下文是不能传播到无事务的stateless session bean)。
有状态会话bean之间调用:此时无论两个有状态会话bean是否支持事务,那么扩展的持久化上下文都会传播,此时就不是通过系统事务来传播的,而是通过statefull session bean的实例来传播(但是此时一定要注意有状态会话bean必须是通过容器注入的或者显示通过JNDI查找)。
无状态的会话bean调用有状态会话bean:一个有事务范围的持久化上下文的stateless session bean调用一个statefull session bean会引发一个错误,因为当前的事务上下文不能传播)
综上所述,EJB之间的持久化上下文传播是通过我们的系统事务来传播的,如果EJB不支持事务(事务属性设置为not support或者never)),那么持久化上下文就不会传播,但是对于扩展的持久化上下文,是通过statefull session bean来传播的,即使没有事务也可以传播,下篇的如何实现会话会再谈到这个问题。<!---->
前面说的是关于持久化如何在持久层的不同的类之间传播的问题,其实无外乎就是通过当前的线程和当前的系统事务来传播,不过对与statefull session bean的扩展的持久化上下文,传播是通过实例来传播的,在下面的实现会话的讨论中,我还会说到这个问题。当我们掌握了原理的时候,遇到一些问题的时候,我们就会很快找到解决方案。
未完待续。。。
下篇“J2EE持久层如何实现会话”
分享到:
相关推荐
### J2EE课程总结 #### 数据库:Oracle **1. Oracle SQL基础知识** - **选择行**:通过`SELECT`语句结合`WHERE`子句来实现特定条件下的数据筛选。 - **限制选择行**:利用`LIMIT`或`ROWNUM`来限制返回结果的数量...
中文译名:J2EE设计模式 作者: (美)William Crawford & Jonathan Kaplan <br/>本书一边在持久化和消息通信等领域中描述一些新模式,一边向你说明如何在J2EE应用的上下文中实现许多模式。有助于考虑问题...
了解实体类、持久化上下文、查询语言(JPQL)是JPA学习的重点。 10. **Web服务**:包括SOAP(Simple Object Access Protocol)和RESTful风格的服务。考生需要知道如何创建和调用Web服务,以及WSDL(Web Services ...
根据提供的标题、描述和部分上下文内容,我们可以推断出这份文档主要介绍了一些与Java 2 Platform, Enterprise Edition(简称 J2EE)相关的经典开发案例。J2EE 是一种广泛应用于企业级应用开发的技术框架,它基于...
11. **CDI**:Contexts and Dependency Injection,用于依赖注入和上下文管理。 12. **JSR-352 Batch**:批处理规范,处理批量任务。 **六、ORM框架——Hibernate** Hibernate是一种流行的ORM框架,它提供了一种在...
考生需理解实体、持久化、查询语言(JPQL)和事务管理。 7. **Web服务(Web Services)**:包括SOAP、WSDL和UDDI,是J2EE中用于跨系统通信的重要部分。考生需要知道如何创建、发布和消费Web服务。 8. **Struts、...
8. **CDI(Contexts and Dependency Injection)**:CDI是Java EE中的依赖注入框架,为Java应用提供了统一的上下文管理和依赖注入服务,与Spring DI类似但更符合Java EE标准。 9. **JMS(Java Message Service)**...
这些知识点涵盖了J2EE开发中的关键概念,包括MVC框架、JSP内置对象、JavaBean规范、Hibernate持久化、Spring容器和依赖注入、Struts2框架以及相关的配置和设计模式。理解和掌握这些知识点对于J2EE开发者来说至关重要...
- **自动持久化**:通过JPA(Java Persistence API)引入,提供了一种统一的持久化模型,使得对象可以自动保存到数据库。 ### 2. JPA (Java Persistence API) JPA是EJB 3.0的一部分,用于管理对象-关系映射(ORM...
- 了解目录服务和命名上下文的概念。 7. **Java Messaging Service (JMS)** - JMS 提供了与消息中间件通信的标准接口。 - 队列和主题的概念,同步和异步消息传递。 - 创建和消费消息的示例。 8. **Java ...
11. **CDI(Contexts and Dependency Injection)**:CDI是Java EE的一项核心功能,它提供了依赖注入和上下文管理,帮助开发者更轻松地管理对象的生命周期和依赖关系。 以上这些概念构成了J2EE平台的基础,理解并...
8. **CDI(Contexts and Dependency Injection)**:CDI为Java应用提供依赖注入和上下文管理,增强了组件的可测试性和可维护性。 9. **WAS(WebSphere Application Server)和JBOSS**:这些是常见的J2EE应用服务器...
7、JNDI的开发主要包括以下步骤:导入JNDI库、创建初始上下文、查找和绑定对象。以下是一段简单的Java代码示例: ```java InitialContext ic = new InitialContext(); DataSource ds = (DataSource) ic.lookup("java...
- **Hibernate**:一个持久层框架,主要用于实现对象关系映射(ORM),简化数据库操作。 - **Struts**:一个基于MVC架构的Web应用程序框架,适用于构建动态的Web应用程序。 掌握主流框架对于J2EE开发者来说至关...
无状态bean适用于不需维护用户会话状态的场景,而有状态bean能保存客户上下文信息。 - **实体bean**:代表业务对象,通常映射到数据库表。有容器管理持久化(CMP)和Bean管理持久化(BMP)两种方式。CMP由容器负责...
2. **EJB(Enterprise JavaBeans)**:EJB是J2EE平台中的核心组件,包括会话bean、实体bean和消息驱动bean,它们分别负责业务逻辑、持久化存储和异步消息处理。 3. **JMS(Java Message Service)**:JMS是Java平台...
1. **Hibernate**:作为JBPM的核心持久层框架,用于处理与数据库之间的交互。 2. **JBPM核心库**: - *bsh.jar*:BeanShell脚本引擎,用于执行流程中的动态脚本。 - *jcr-1.0.jar*:内容存储库接口库,JBPM中用于...
笔记会涵盖实体、持久化上下文和查询语言(JPQL)。 8. **MVC(Model-View-Controller)**架构:这是Web开发中常见的设计模式,笔记会讲解如何在J2EE中实现。 9. **部署描述符**:如web.xml和ejb-jar.xml,是配置...
2. **Spring框架**:Spring作为应用上下文和依赖注入(DI)框架,用于管理对象的生命周期和配置。它还提供了AOP(面向切面编程)支持,实现事务管理、安全控制等功能。在进销存系统中,Spring可以用来管理数据库连接...