`

hibernate-事务管理

阅读更多
Hibernate 是JDBC 的轻量级封装,本身并不具备事务管理能力。在事务管理层,
Hibernate将其委托给底层的JDBC或者JTA,以实现事务管理和调度功能。
Hibernate的默认事务处理机制基于JDBC Transaction。我们也可以通过配置文
件设定采用JTA作为事务管理实现:
<hibernate-configuration>
<session-factory>
……
<property name="hibernate.transaction.factory_class">
net.sf.hibernate.transaction.JTATransactionFactory
<!--net.sf.hibernate.transaction.JDBCTransactionFactory-->
</property>
……
</session-factory>
</hibernate-configuration>

基于JDBC的事务管理将事务管理委托给JDBC 进行处理无疑是最简单的实现方式,Hibernate 对于JDBC事务的封装也极为简单。
我们来看下面这段代码:
session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
……
tx.commit();

从JDBC层面而言,上面的代码实际上对应着:
Connection dbconn = getConnection();
dbconn.setAutoCommit(false);
……
dbconn.commit();

就是这么简单,Hibernate并没有做更多的事情(实际上也没法做更多的事情),只是将这样的JDBC代码进行了封装而已。
这里要注意的是,在sessionFactory.openSession()中,hibernate会初始化数据库连接,与此同时,将其AutoCommit 设为关闭状态(false)。而其后,在Session.beginTransaction 方法中,Hibernate 会再次确认Connection 的AutoCommit 属性被设为关闭状态( 为了防止用户代码对session 的Connection.AutoCommit属性进行修改)。
这也就是说,我们一开始从SessionFactory获得的session,其自动提交属性就已经被关闭(AutoCommit=false),下面的代码将不会对数据库产生任何效果:
session = sessionFactory.openSession();
session.save(user);
session.close();

这实际上相当于 JDBC Connection的AutoCommit属性被设为false,执行了若干JDBC操作之后,没有调用commit操作即将Connection关闭。如果要使代码真正作用到数据库,我们必须显式的调用Transaction指令:
session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
session.save(user);
tx.commit();
session.close();

基于JTA的事务管理
JTA 提供了跨Session 的事务管理能力。这一点是与JDBC Transaction 最大的差异。
JDBC事务由Connnection管理,也就是说,事务管理实际上是在JDBC Connection中实现。事务周期限于Connection的生命周期之类。同样,对于基于JDBC Transaction的Hibernate 事务管理机制而言,事务管理在Session 所依托的JDBC Connection中实现,事务周期限于Session的生命周期。
JTA 事务管理则由 JTA 容器实现,JTA 容器对当前加入事务的众多Connection 进
行调度,实现其事务性要求。JTA的事务周期可横跨多个JDBC Connection生命周期。
同样对于基于JTA事务的Hibernate而言,JTA事务横跨可横跨多个Session。
JTA 事务是由JTA Container 维护,而参与事务的Connection无需对事务管理进行干涉。这也就是说,如果采用JTA Transaction,我们不应该再调用HibernateTransaction功能。
上面基于JDBC Transaction的正确代码,这里就会产生问题:
public class ClassA{
public void saveUser(User user){
session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
session.save(user);
tx.commit();
session.close();
}
}
public class ClassB{
public void saveOrder(Order order){
session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
session.save(order);
tx.commit();
session.close();
}
}
public class ClassC{
public void save(){
……
UserTransaction tx = new InitialContext().lookup(“……”);
ClassA.save(user);
ClassB.save(order);
tx.commit();
……
}
}
这里有两个类ClassA和ClassB,分别提供了两个方法:saveUsersaveOrder,
用于保存用户信息和订单信息。在ClassC中,我们接连调用了ClassA.saveUser方法和ClassB.saveOrder 方法,同时引入了JTA 中的UserTransaction 以实现ClassC.save方法中的事务性。问题出现了,ClassA 和ClassB 中分别都调用了Hibernate 的Transaction 功能。在Hibernate 的JTA 封装中,Session.beginTransaction 同样也执行了InitialContext.lookup方法获取UserTransaction实例,Transaction.commit方法同样也调用了UserTransaction.commit方法。实际上,这就形成了两个嵌套式的JTA Transaction:ClassC 申明了一个事务,而在ClassC 事务周期内,ClassA 和ClassB也企图申明自己的事务,这将导致运行期错误。因此,如果决定采用JTA Transaction,应避免再重复调用Hibernate 的
Transaction功能,上面的代码修改如下:
public class ClassA{
public void save(TUser user){
session = sessionFactory.openSession();
session.save(user);
session.close();
}
……
}
public class ClassB{
public void save (Order order){
session = sessionFactory.openSession();
session.save(order);
session.close();
}
……
}
public class ClassC{
public void save(){
……
UserTransaction tx = new InitialContext().lookup(“……”);
classA.save(user);
classB.save(order);
tx.commit();
……
}
}

上面代码中的ClassC.save方法,也可以改成这样:
public class ClassC{
public void save(){
……
session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
classA.save(user);
classB.save(order);
tx.commit();
……
}
}

实际上,这是利用Hibernate来完成启动和提交UserTransaction的功能,但这样的做法比原本直接通过InitialContext获取UserTransaction 的做法消耗了更多的资源,得不偿失。
在EJB 中使用JTA Transaction 无疑最为简便,我们只需要将save 方法配置为JTA事务支持即可,无需显式申明任何事务,下面是一个Session Bean的save方法,它的事务属性被申明为“Required”,EJB容器将自动维护此方法执行过程中的事务:
/**
* @ejb.interface-method
* view-type="remote"
*
* @ejb.transaction type = "Required"
**/
public void save(){
//EJB环境中,通过部署配置即可实现事务申明,而无需显式调用事务
classA.save(user);
classB.save(log);
}//方法结束时,如果没有异常发生,则事务由EJB容器自动提交。
分享到:
评论
7 楼 nanjiwubing123 2012-04-26  
不错不错
6 楼 junj 2008-12-29  
业务层依赖于DAO层,DAO只服务于业务层,这种情况下,你至对业务层上进行各种事务配置,就是你现在这种场景。其实DAO层开启事务控制可以减少资源泄漏和逻辑上的错误
5 楼 wangyugod 2008-12-27  
事务应该在业务逻辑层控制,建议使用拦截器做事务控制,并且消除掉丑陋的DAO层
4 楼 zhouyaguo 2008-12-27  
Joo说的没错,我的项目中都是在具体的业务逻辑层进行事务控制,而把Hibernate仅当作对Jdbc的简单封装,不进行事务处理,这样做,对于要换成JTA有好处,对我们的业务层事务控制层次也清晰。
不过要保证事务使用的session和DAO里的session是同一个。
3 楼 yangdefeng95802 2008-06-12  
同意LZ.
2 楼 cuiguodong 2008-06-12  
同意楼上
1 楼 Joo 2008-04-12  
我一直不明白为什么要在Hibernate层做事务呢,难道不应该把事务提高到业务逻辑层么?
因为一般来说Hibernate的DAO层都是比较细颗粒的操作
但是到逻辑层了就需要把这些细颗粒的操作组合起来形成事务,如果你在Hibernate这里做一次,到业务层又需要事务管理怎么办

相关推荐

    hibernate-release-4.1.4

    【标题】"hibernate-release-4.1.4" 是Hibernate...通过深入研究这个压缩包,开发者不仅可以了解Hibernate的基本用法,还能掌握更高级的功能,如事务管理、缓存策略、查询语言(HQL)等,从而提升开发效率和代码质量。

    hibernate-release-5.2.10

    5. **事务管理**:Hibernate支持编程式和声明式事务管理,确保数据的一致性。 6. **缓存**:Hibernate内置了二级缓存机制,可以通过配置使用如Ehcache这样的缓存提供者,提高性能。 7. **关联映射**:包括一对一(@...

    hibernate-release-5.0.7.Final的所有jar包

    2. **hibernate-entitymanager**: 用于支持JPA规范,提供实体管理和事务处理。如果你的应用程序需要遵循JPA标准,这个jar包是必不可少的。 3. **hibernate-jpa-2.1-api**: 提供JPA 2.1的API接口,是Hibernate与JPA...

    hibernate-core 核心jar包

    7. **事务处理**:Hibernate支持JTA(Java Transaction API)和JDBC事务,提供了事务的开始、提交、回滚等操作,确保数据的一致性和完整性。 8. **级联操作**:在映射文件中,可以配置实体属性的级联行为,如保存、...

    hibernate-core-5.0.11.Final.jar

    3. **事务管理**:使用Session的`beginTransaction()`、`commit()`和`rollback()`方法进行事务控制。 4. **查询数据**:使用HQL或Criteria API进行复杂的数据检索。 六、性能优化 - 使用二级缓存提高性能,如...

    hibernate-jpa-2.1-api-1.0.0.final.jar.zip

    - **事务管理**: JPA 2.1提供了@TransactionAttribute注解来控制事务的边界,配合EntityManager的flush()和clear()方法进行事务操作。 **4. 性能优化** - **缓存**: Hibernate的二级缓存可以显著提升性能,通过...

    最新 hibernate-release-4.2.13.Final.tgz

    4. 事务管理:了解Hibernate的事务处理策略,包括自动提交、显式事务控制等。 总结: "hibernate-release-4.2.13.Final.tgz"是一个全面的Hibernate学习资源,不仅提供了运行环境所需的jar包,还包含了丰富的实践...

    hibernate-release-5.0.0.Final(1).zip

    4. Transaction:Hibernate支持事务管理,确保数据的一致性和完整性。 四、实战应用 在实际项目中,我们可以利用Hibernate简化数据库操作,例如: - 使用SessionFactory创建Session实例。 - 通过Session的save()或...

    hibernate-release-4.3.1.Final.zip

    Hibernate在这里作为数据访问层,需要与其他jar包配合工作,比如Spring用于管理和协调业务逻辑,Struts则处理前端展示。"hibernate3.jar"是Hibernate 3.x系列的主库,而"hibernate-jpa-2.1-api-1.0.0.Final.jar"则是...

    Hibernate-annotations-3.4最新版本

    在实际应用中,Hibernate-annotations-3.4.0.GA还包含了事务管理、缓存机制等功能。@Transactional注解可以标记一个方法为事务边界,而Hibernate的二级缓存通过@Cacheable和@Cache配置,可以提高查询性能,降低...

    hibernate-release-5.3.2.Final

    1. Session接口:作为与数据库交互的主要接口,Session负责保存、更新、删除和检索对象,支持事务管理和并发控制。 2. Criteria API:提供了一种比HQL更面向对象的查询方式,可以动态构建查询条件,避免硬编码SQL...

    hibernate-release-4.2.4.Final.zip

    9. **事务管理**:Hibernate支持自动和手动的事务管理,可以很好地处理并发和数据一致性问题。开发者可以通过Session的beginTransaction、commit和rollback方法来控制事务的边界。 10. **缓存机制**:Hibernate提供...

    hibernate-release-5.0.7.Final.zip

    - Transaction:Hibernate提供了基于JTA和JDBC的事务管理,确保数据的一致性。 - Criteria API:提供了一种更灵活、更类型安全的方式来执行查询,替代了传统的HQL(Hibernate Query Language)。 - Object-...

    hibernate-cglib-repack-2.1_3.jar

    但在Hibernate中,CGLIB主要用于动态代理,生成实体类的子类,从而实现在不修改原有代码的情况下,添加额外的行为,如懒加载、事务管理等。 在Hibernate中,实体类通常会被CGLIB动态代理,生成一个代理对象。当调用...

    hibernate-release-5.0.7 全包

    9. **geronimo-jta_1.1_spec-1.1.1.jar**:这是一个JTA(Java Transaction API)的规范实现,用于支持分布式事务处理,是Hibernate进行事务管理的基础。 在实际开发中,这些组件共同协作,使得开发者可以通过简单的...

    hibernate-release-4.2.4核心jar包

    最后,`jboss-transaction-api_1.1_spec-1.0.1.Final.jar`包含了Java EE 6的事务API规范,提供了事务管理和协调的功能,这对于在分布式环境中保证数据的一致性至关重要。 总结来说,`hibernate-release-4.2.4_jar....

    hibernate-distribution-3.6.2 API及jar包

    这份PDF文档详细介绍了Hibernate的核心概念、配置、实体映射、查询语言(HQL)以及事务和缓存管理等内容。通过阅读这份文档,开发者可以了解到如何将Java对象与数据库表进行映射,实现数据的持久化,从而避免直接...

    hibernate-orm-master.zip

    你可以通过阅读源码来理解Hibernate的内部工作机制,比如实体管理、会话管理、缓存策略、事务处理、查询语言(HQL)和Criteria API等方面。 1. **实体管理**:Hibernate通过@Entity注解将Java类声明为数据库中的表,...

    hibernate-extensions-2.1.3.zip ,middlegen for hibernate

    - Cache管理:提供更细粒度的缓存控制,包括第二级缓存和查询缓存的优化。 - Batch处理:支持批量插入、更新和删除操作,提高大数据量操作的性能。 - Event监听器:允许在特定的持久化操作前后执行自定义逻辑,如...

    hibernate-distribution-3.3.2.GA-dist

    同时,理解和熟练运用Hibernate的异常体系、事务管理以及查询优化,也是提升开发效率的关键。 总的来说,Hibernate 3.3.2.GA版本为Java开发者提供了强大的ORM工具,通过其丰富的特性和良好的社区支持,使得数据库...

Global site tag (gtag.js) - Google Analytics