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>

1、基于JDBC的事务管理

将事务管理委托给JDBC 进行处理无疑是最简单的实现方式,Hibernate 对于JDBC事务的封装也极为简单:

session = sessionFactory.openSession();
    Transaction tx = session.beginTransaction();
     ……
     tx.commit();

在sessionFactory.openSession()中,Hibernate会初始化数据库连接,与此同时,将其 AutoCommit设为关闭状态,这就是说,从SessionFactory获得session,其自动提交属性就已经被关闭了,下面的代码不会对事务性数据库产生任何效果。

session=sessionFactory.openSession();

session.save(user);

session.close();

如果要使得代码真正作用到数据库,必须显示的调用Transaction指令

session=sessionFactory.openSession();

Transaction tx = session.beginTransaction();

session.save(user);

tx.commit();

session.close();

从JDBC层面而言,上面的代码实际上对应着:

Connection dbconn = getConnection();
    dbconn.setAutoCommit(false);
    ……
    dbconn.commit();

Hibernate并没有做更多的事情(实际上也没法做更多的事情),只是将这样的JDBC代码进行了封装而已。

2、基于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容器自动提交。

时间: 2024-08-21 15:14:11

Hibernate事务管理的相关文章

Spring与Hibernate整合,实现Hibernate事务管理

1.所需的jar包 连接池/数据库驱动包 Hibernate相关jar Spring 核心包(5个) Spring aop 包(4个) spring-orm-3.2.5.RELEASE.jar                 [spring对hibernate的支持] spring-tx-3.2.5.RELEASE.jar                     [事务相关] 2.配置文件 Product.hbm.xml <?xml version="1.0" encoding=

Spring对Hibernate事务管理【转】

在谈Spring事务管理之前我们想一下在我们不用Spring的时候,在Hibernate中我们是怎么进行数据操作的.在Hibernate中我们每次进行一个操作的的时候我们都是要先开启事务,然后进行数据操作,然后提交事务,关闭事务,我们这样做的原因是因为Hibernate默认的事务自动提交是false,他是需要我们人为的手动提交事务,假如你不想每次都手动提交事务的话,你可以在hibernate.cfg.xml我文件中把它设置为事务自动提交: <property name="hibernate

Hibernate事务管理-HibernateTransactionManager-对hibernate session的管理

由于对SSH还停留在比较初级的应用水平上,今天在遇到一个疑惑时折腾了很久,具体问题是这样的, 有这么一个测试方法, 1 public static void test1() { 2 ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml","daoContext.xml"); 3 MgrManager mgr = (MgrManager)ctx.getBe

Spring对Hibernate事务管理

谈Spring事务管理之前我们想一下在我们不用Spring的时候,在Hibernate中我们是怎么进行数据操作的.在Hibernate中 我们每次进行一个操作的的时候我们都是要先开启事务,然后进行数据操作,然后提交事务,关闭事务,我们这样做的原因是因为Hibernate默认的事务自 动提交是false,他是需要我们人为的手动提交事务,假如你不想每次都手动提交事务的话,你可以在hibernate.cfg.xml我文件中把它设置 为事务自动提交: <property name="hiberna

Hibernate 事务管理

一. 事务包含四个基本特性:简称ACID: 1. Atomic(原子性):全部成功或全部失败: 2. Consistency(一致性):只有合法数据才能被写入,不合法则回滚到最初状态: 3. Isolation(隔离性):允许并发,并发的事务相互独立: 4. Durability(持久性):事务结束后,结果能保存: 二. 数据库事务管理隔离等级 数据库操作过程中经常出现三种不确定情况: 1. 脏读取(Dirty Reads):一个事务读取了另一个并行事务未提交的数据: 2. 不可重复读取(Non

Spring中配置Hibernate事务管理

<!-- transationManager --> <bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory"/> </bean>

[转]Spring的事务管理难点剖析(1):DAO和事务管理的牵绊

原文地址:http://stamen.iteye.com/blog/1441758 有些人很少使用Spring而不使用Spring事务管理器的应用,因此常常有人会问:是否用了Spring,就一定要用Spring事务管理器,否则就无法进行数据的持久化操作呢?事务管理器和DAO是什么关系呢?   也许是DAO和事务管理如影随行的缘故吧,这个看似简单的问题实实在在地存在着,从初学者心中涌出,萦绕在老手的脑际.答案当然是否定的!我们都知道:事 务管理是保证数据操作的事务性(即原子性.一致性.隔离性.持久

Spring学习8-Spring事务管理(编程式事务管理)

一.Spring事务的相关知识   1.事务是指一系列独立的操作,但在概念上具有原子性. 比如转账:A账号-100, B账号+100,完成.这两个操作独立是没问题的. 但在逻辑上,要么全部完成,要么一起失败.    1)jdbc事务:每个Connection都带有一个事务,只是默认被设置为自动提交.一个连接可以有多个事务.对于JDBC,只有在同一个连接内,才有讨论是否提交的前提. 2)Hibernate事务:本质上也是使用JDBC来处理事务.但是不是直接操作,而是使用Session来操作事务.S

Spring 事务管理高级应用难点剖析--转

第 1 部分 http://www.ibm.com/search/csass/search/?q=%E4%BA%8B%E5%8A%A1&sn=dw&lang=zh&cc=CN&en=utf&hpp=20&dws=cndw&lo=zh 概述 Spring 最成功,最吸引人的地方莫过于轻量级的声明式事务管理,仅此一点,它就宣告了重量级 EJB 容器的覆灭.Spring 声明式事务管理将开发者从繁复的事务管理代码中解脱出来,专注于业务逻辑的开发上,这是一件