hibernate-事务管理

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

Hibernate将其委托给底层的JDBC或者JTA,以实现事务管理和调度功能。

Hibernate的默认事务处理机制基于JDBC Transaction。我们也可以通过配置文

件设定采用JTA作为事务管理实现:

Java代码

……

net.sf.hibernate.transaction.JTATransactionFactory

……

……net.sf.hibernate.transaction.JTATransactionFactory……

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

我们来看下面这段代码:

Java代码

session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

……

tx.commit();

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

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

Java代码

Connection dbconn = getConnection();

dbconn.setAutoCommit(false);

……

dbconn.commit();

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),下面的代码将不会对数据库产生任何效果:

Java代码

session = sessionFactory.openSession();

session.save(user);

session.close();

session = sessionFactory.openSession();session.save(user);session.close();

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

Java代码

session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

session.save(user);

tx.commit();

session.close();

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的正确代码,这里就会产生问题:

Java代码

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();

……

}

}

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功能,上面的代码修改如下:

Java代码

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();

……

}

}

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方法,也可以改成这样:

Java代码

public class ClassC{

public void save(){

……

session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

classA.save(user);

classB.save(order);

tx.commit();

……

}

}

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容器将自动维护此方法执行过程中的事务:

Java代码

/**

* @ejb.interface-method

* view-type="remote"

*

* @ejb.transaction type = "Required"

**/

public void save(){

//EJB环境中,通过部署配置即可实现事务申明,而无需显式调用事务

classA.save(user);

classB.save(log);

}//方法结束时,如果没有异常发生,则事务由EJB容器自动提交。

/*** @ejb.interface-method* view-type="remote"** @ejb.transaction type = "Required"**/public void save(){//EJB环境中,通过部署配置即可实现事务申明,而无需显式调用事务classA.save(user);classB.save(log);}//方法结束时,如果没有异常发生,则事务由EJB容器自动提交。

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

Hibernate将其委托给底层的JDBC或者JTA,以实现事务管理和调度功能。

Hibernate的默认事务处理机制基于JDBC Transaction。我们也可以通过配置文

件设定采用JTA作为事务管理实现:

Java代码

……

net.sf.hibernate.transaction.JTATransactionFactory

……

……net.sf.hibernate.transaction.JTATransactionFactory……

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

我们来看下面这段代码:

Java代码

session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

……

tx.commit();

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

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

Java代码

Connection dbconn = getConnection();

dbconn.setAutoCommit(false);

……

dbconn.commit();

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),下面的代码将不会对数据库产生任何效果:

Java代码

session = sessionFactory.openSession();

session.save(user);

session.close();

session = sessionFactory.openSession();session.save(user);session.close();

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

Java代码

session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

session.save(user);

tx.commit();

session.close();

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的正确代码,这里就会产生问题:

Java代码

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();

……

}

}

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功能,上面的代码修改如下:

Java代码

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();

……

}

}

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方法,也可以改成这样:

Java代码

public class ClassC{

public void save(){

……

session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

classA.save(user);

classB.save(order);

tx.commit();

……

}

}

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容器将自动维护此方法执行过程中的事务:

Java代码

/**

* @ejb.interface-method

* view-type="remote"

*

* @ejb.transaction type = "Required"

**/

public void save(){

//EJB环境中,通过部署配置即可实现事务申明,而无需显式调用事务

classA.save(user);

classB.save(log);

}//方法结束时,如果没有异常发生,则事务由EJB容器自动提交。

/*** @ejb.interface-method* view-type="remote"** @ejb.transaction type = "Required"**/public void save(){//EJB环境中,通过部署配置即可实现事务申明,而无需显式调用事务classA.save(user);classB.save(log);}//方法结束时,如果没有异常发生,则事务由EJB容器自动提交。


相关文章

  • 三大框架的作用
  • Struts2的作用 作为web层框架应该大体实现以下功能: 1.获取表单内容,并组织生成参数对象 2.根据请求的参数转发请求给适当的控制器 3.在控制器中调用业务接口 4.将业务接口返回的结果包装起来发送给指定的视图,并由视图完成处理结果 ...查看


  • 项目经验参考写法
  • 2007/03至2009/02 java 程序员 XXXX 有限公司 项目经验 项目名称:寰宇电子商务网站 开发工具:SQL Server2005+MyEclipse6.5+Tomcat 运用技术:Jsp+Struts+Spring+Hib ...查看


  • 小型购物系统需求分析
  • 1. 引言 1.1编写目的 2. 综合描述 2.1产品的技术 2.2产品的功能 2.3开发及运行环境 3. 系统功能需求 3.1系统需求分析 3.2系统流程图 4. 其他功能需求 4.1性能需求 4.2 开放性要求 4.3 安全性要求 5. ...查看


  • Hibernate入门之自己写的小例子的总结-JB--IT之博客
  • Hibernate入门之自己写的小例子的总结-JB- - 作者:vatful 整理人:smallduzi 版权声明:本文严禁转载,如有转载请求,请和作者联系 本机环境介绍:JB7.0, hibernate-2.0.3, Oracle8.1客 ...查看


  • javaWeb开发文献综述
  • 毕业设计(论文)文献综述 基于JAVA 的火车售票系统的设计 学生姓名 周誉 学 号 [1**********]5 指导教师 柳斌 何剑锋 职称 副教授 专 业 软件工程 二零一五年六月 摘要: 随着我国人口的增长及社会的飞速发展,窗口售票 ...查看


  • 计算机专业简历大全
  • 简历模板二 求职简历 基本信息 姓 名: 出生日期: 工作年限: 1985年9月12日 一年以上 性 别: 居 住 地: 电子邮件: 专 业: QQ 号码: 男 北京-海淀区 软件开发 学 历: 本科 (四年) 手机号码: 详细信息 ◆ 自 ...查看


  • 小区车辆管理系统的设计与实现
  • 小区车辆管理系统的设计与实现 专业: 学生: 指导老师: 摘 要 小区车辆管理系统是一个通过人机交互系统,将小区车辆.驾驶员和出车情况进行存储,并同时对小区车辆进行全面管理的系统.它主要实现了车辆信息管理.驾驶员信息管理.出车信息管理等功能 ...查看


  • 基于MVC的高校科研成果管理系统的设计与实现
  • 第21卷第10期2011年10月 计算机技术与发展 COMPUTER TECHNOLOGY AND DEVELOPMENT Vol.21No.10Oct.2011 基于MVC 的高校科研成果管理系统 的设计与实现 梁 弼 (四川文理学院计算 ...查看


  • Java软件工程师简历模板
  • XX个人简历 个人信息 姓 名: 出生日期: 工作年限: 手 机: 户 口: 婚姻状况: 关 键 词: XXX 1986年2月16日 三年以上 性 别: 居 住 地: 电子邮件: 上海-浦东区 [email protected] 086- [1* ...查看


热门内容