zoukankan      html  css  js  c++  java
  • 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容器自动提交。

  • 相关阅读:
    【转】 史上最详尽的平衡树(splay)讲解与模板(非指针版spaly)
    HihoCoder1325 : 平衡树·Treap(附STL版本)
    HihoCOder1323 : 回文字符串(区间DP)
    从分布式一致性算法到区块链共识机制
    Nacos Committer 张龙:Nacos Sync 的设计原理和规划
    MaxCompute Studio使用心得系列7——作业对比
    阿里云MaxCompute 2019-4月刊
    DDoS攻击新趋势:海量移动设备成为新一代肉鸡
    胡喜:从 BASIC 到 basic ,蚂蚁金服技术要解决两个基本的计算问题
    阿里开发者招聘节 | 面试题14:如何实现两金额数据相加(最多小数点两位)
  • 原文地址:https://www.cnblogs.com/moonandstar08/p/5143633.html
Copyright © 2011-2022 走看看