(十)Hibernate之事务管理

事务的定义

事务就是指作为单个逻辑工作单元执行的一组数据操作,这些操作要么必须全部成功,要么必须全部失败,以保证数据的一致性和完整性。

事务具有ACID属性

 

 原子性(Atomic):事务由一个或多个行为绑在一起组成,好像是一个单独的工作单元。原子性确保在事务中的所有操作要么都发生,要么都不发生。

 一致性(Consistent):一旦一个事务结束了(不管成功与否),系统所处的状态和它的业务规则是一致的。即数据应当不会被破坏。

 隔离性(Isolated):事务应该允许多个用户操作同一个数据,一个用户的操作不会和其他用户的操作相混淆。

 持久性(Durable):一旦事务完成,事务的结果应该持久化。

 


 

 事务的ACID特性是由关系数据库管理系统(RDBMS)来实现的。

1)数据库管理系统采用日志来保证事务的原子性、一致性和持久性。日志记录了事务对数据库所做的更新,如果某个事务在执行过程中发生错误,就可以根据日志,撤销事务对数据库已做的更新,使数据库退回到执行事务前的初始状态。

2)数据库管理系统采用锁机制来实现事务的隔离性。当多个事务同时更新数据库相同的数据时,只允许持有锁的事务能更新该数据,其他事务必须等待,直到前一个事务释放了锁,其他事务才有机会更新该数据。

 


数据库事务声明

数据库系统的客户程序只要向数据库系统声明了一个事务,数据库系统就会自动保证事务的ACID特性。在JDBC API中,java.sql.Connection类代表一个数据库连接。它提供了以下方法控制事务:

1.   setAutoCommit(Boolean autoCommit):设置是否自动提交事务。

2.    commit():提交事务。

3.     rollback():撤销事务 

JDBC API声明事务的示例代码如下:

Java代码   收藏代码
  1. <span style="font-size: large;">Connection = null;  
  2. PreparedStatement pstmt = null;  
  3. try{  
  4. con = DriverManager.getConnection(dbUrl, username, password);  
  5. //设置手工提交事务模式  
  6. con.setAutoCommit(false);  
  7. pstmt = ……;  
  8. pstmt.executeUpdate();  
  9. //提交事务  
  10. con.commit();  
  11. }catch(Exception e){  
  12. //事务回滚  
  13. con.rollback();  
  14. …..  
  15. } finally{  
  16.     …….  
  17. }</span>  
 

Hibernate 是JDBC 的轻量级封装,本身并不具备事务管理能力。

在事务管理层, Hibernate将其委托给底层的JDBC或者JTA,只是将底层的JDBCTransaction或者JTATransaction进行封装一下,在外边套上Transaction和Session的外壳,以实现事务管理和调度功能。

1)Hibernate中使用JDBC事务
如果在Hibernate中使用JDBC事务,可以在hibernate.cfg.xml中指定Hibernate事务为JDBCTransaction。如果不进行配置,Hibernate会默认使用JDBC事务。代码如下


Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;"><?xml version="1.0" encoding="utf-8" ?>  
  2. <!DOCTYPE hibernate-configuration PUBLIC  
  3.     "-//Hibernate/Hibernate Configuration DTD 3.0//EN"  
  4.     "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">  
  5. <hibernate-configuration>  
  6. <session-factory>  
  7.       <property name="hibernate.transaction.factory_class">      
  8.       org.hibernate.transaction.JDBCTransactionFactory  
  9.       </property>      
  10. </session-factory>  
  11. </hibernate-configuration></span></span>  

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


我们来看下面这段代码:

Java代码
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">Transaction tx=null;  
  2. try{  
  3. session = sessionFactory.openSession();      
  4. Transaction tx = session.beginTransaction();    //开启事务  
  5. //执行持久化操作  
  6. ……      
  7. tx.commit();   //操作正常,提交事务  
  8. }catch(RuntimeException e){  
  9.  if(tx!=null){  
  10.  tx.rollback();//操作过程中有一场,回滚事务  
  11.  throw e;//处理异常  
  12. }  
  13. }finally{  
  14.  session.close()  
  15. } </span></span>  
  从JDBC层面而言,上面的代码实际上对应着: 
Java代码
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">Connection dbconn = getConnection();      
  2. dbconn.setAutoCommit(false);      
  3. ……      
  4. dbconn.commit();    </span></span>  
  就是这么简单,Hibernate并没有做更多的事情(实际上也没法做更多的事情),只是将这样的JDBC代码进行了封装而已。 
这里要注意的是,在sessionFactory.openSession()中,hibernate会初始化数据库连接,与此同时,将其AutoCommit 设为关闭状态(false)。而其后,在Session.beginTransaction 方法中,Hibernate 会再次确认Connection 的AutoCommit 属性被设为关闭状态( 为了防止用户代码对session 的Connection.AutoCommit属性进行修改)。
这也就是说,我们一开始从SessionFactory获得的session,其自动提交属性就已经被关闭(AutoCommit=false),下面的代码将不会对数据库产生任何效果:
Java代码
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">session = sessionFactory.openSession();      
  2. session.save(user);      
  3. session.close();     
  4. session = sessionFactory.openSession();  
  5. session.save(user);  
  6. session.close(); </span></span>  
  这实际上相当于 JDBC Connection的AutoCommit属性被设为false,执行了若干JDBC操作之后,没有调用commit操作即将Connection关闭。如果要使代码真正作用到数据库,我们必须显式的调用Transaction指令: 
Java代码
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">session = sessionFactory.openSession();      
  2. Transaction tx = session.beginTransaction();      
  3. session.save(user);      
  4. tx.commit();      
  5. session.close();    </span></span>  
2)  Hibernate中使用JTA事务
具体配置为
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;"><?xml version="1.0" encoding="utf-8" ?>  
  2. <!DOCTYPE hibernate-configuration PUBLIC  
  3.     "-//Hibernate/Hibernate Configuration DTD 3.0//EN"  
  4.     "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">  
  5. <hibernate-configuration>  
  6. <session-factory>  
  7.       <property name="hibernate.transaction.factory_class">      
  8.       org.hibernate.transaction.JTATransactionFactory  
  9.       </property>      
  10. </session-factory>  
  11. </hibernate-configuration></span></span>  
 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的正确代码,这里就会产生问题:

Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">public class ClassA{      
  2. public void saveUser(User user){      
  3. session = sessionFactory.openSession();      
  4. Transaction tx = session.beginTransaction();      
  5. session.save(user);      
  6. tx.commit();      
  7. session.close();      
  8. }      
  9. }      
  10. public class ClassB{      
  11. public void saveOrder(Order order){      
  12. session = sessionFactory.openSession();      
  13. Transaction tx = session.beginTransaction();      
  14. session.save(order);      
  15. tx.commit();      
  16. session.close();      
  17. }      
  18. }      
  19. public class ClassC{      
  20. public void save(){      
  21. ……      
  22. UserTransaction tx = new InitialContext().lookup(“……”);      
  23. ClassA.save(user);      
  24. ClassB.save(order);      
  25. tx.commit();      
  26. ……      
  27. }      
  28. }     
  29. public class ClassA{  
  30. public void saveUser(User user){  
  31. session = sessionFactory.openSession();  
  32. Transaction tx = session.beginTransaction();  
  33. session.save(user);  
  34. tx.commit();  
  35. session.close();  
  36. }  
  37. }  
  38. public class ClassB{  
  39. public void saveOrder(Order order){  
  40. session = sessionFactory.openSession();  
  41. Transaction tx = session.beginTransaction();  
  42. session.save(order);  
  43. tx.commit();  
  44. session.close();  
  45. }  
  46. }  
  47. public class ClassC{  
  48. public void save(){  
  49. try{  
  50. UserTransaction tx = (UserTransaction)new InitialContext().lookup(“java:comp/UserTransaction”);  
  51. tx.begin();//开启JTA事务  
  52. ClassA.save(user);  
  53. ClassB.save(order);  
  54. tx.commit();//操作正常,提交JTA事务  
  55. }catch(RuntimeException e){  
  56. tx.rollback();//操作异常,回滚JTA事务  
  57. throw e;//异常处理  
  58. }finally{  
  59. }  
  60. }  
  61. }</span></span>  
    这里有两个类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功能,上面的代码修改如下:
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="white-space: normal; font-size: 12px;"><span style="font-size: large;"><span style="font-size: large;"> </span></span><span style="white-space: pre;"><span style="font-size: large;"><span style="font-size: large;">public class ClassA{</span></span></span></span></span>  
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">public void save(TUser user){      
  2. session = sessionFactory.openSession();      
  3. session.save(user);      
  4. session.close();      
  5. }      
  6. ……      
  7. }      
  8. public class ClassB{      
  9. public void save (Order order){      
  10. session = sessionFactory.openSession();      
  11. session.save(order);      
  12. session.close();      
  13. }      
  14. ……      
  15. }      
  16. public class ClassC{      
  17. public void save(){      
  18. ……      
  19. UserTransaction tx = new InitialContext().lookup(“……”);      
  20. classA.save(user);      
  21. classB.save(order);      
  22. tx.commit();      
  23. ……      
  24. }      
  25. }     
  26. public class ClassA{  
  27. public void save(TUser user){  
  28. session = sessionFactory.openSession();  
  29. session.save(user);  
  30. session.close();  
  31. }  
  32. ……  
  33. }  
  34. public class ClassB{  
  35. public void save (Order order){  
  36. session = sessionFactory.openSession();  
  37. session.save(order);  
  38. session.close();  
  39. }  
  40. ……  
  41. }  
  42. public class ClassC{  
  43. public void save(){  
  44. ……  
  45. UserTransaction tx = new InitialContext().lookup(“……”);  
  46. classA.save(user);  
  47. classB.save(order);  
  48. tx.commit();  
  49. ……  
  50. }  
  51. }  
  52. </span></span>  
 上面代码中的ClassC.save方法,也可以改成这样: 
Java代码
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">public class ClassC{      
  2. public void save(){      
  3. ……      
  4. session = sessionFactory.openSession();      
  5. Transaction tx = session.beginTransaction();      
  6. classA.save(user);      
  7. classB.save(order);      
  8. tx.commit();      
  9. ……      
  10. }      
  11. }     
  12. </span></span>  
  实际上,这是利用Hibernate来完成启动和提交UserTransaction的功能,但这样的做法比原本直接通过InitialContext获取UserTransaction 的做法消耗了更多的资源,得不偿失。 
在EJB 中使用JTA Transaction 无疑最为简便,我们只需要将save 方法配置为JTA事务支持即可,无需显式申明任何事务,下面是一个Session Bean的save方法,它的事务属性被申明为“Required”,EJB容器将自动维护此方法执行过程中的事务:
Java代码
Java代码   收藏代码
  1. <span style="font-size: large;"><span style="font-size: large;">/**    
  2. * @ejb.interface-method    
  3. * view-type="remote"    
  4. *    
  5. * @ejb.transaction type = "Required"    
  6. **/     
  7. public void save(){      
  8. //EJB环境中,通过部署配置即可实现事务申明,而无需显式调用事务      
  9. classA.save(user);      
  10. classB.save(log);      
  11. }//方法结束时,如果没有异常发生,则事务由EJB容器自动提交。</span></span> 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值