`
JavaCrazyer
  • 浏览: 3008717 次
  • 性别: Icon_minigender_1
  • 来自: 河南
社区版块
存档分类

Hibernate温习(10)--应用程序中的事务管理

阅读更多

事务的定义

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

事务具有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声明事务的示例代码如下:

Connection = null;
PreparedStatement pstmt = null;
try{
con = DriverManager.getConnection(dbUrl, username, password);
//设置手工提交事务模式
con.setAutoCommit(false);
pstmt = ……;
pstmt.executeUpdate();
//提交事务
con.commit();
}catch(Exception e){
//事务回滚
con.rollback();
…..
} finally{
    …….
}
 

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

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

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


<?xml version="1.0" encoding="utf-8" ?>
<!DOCTYPE hibernate-configuration PUBLIC
	"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
	"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
	  <property name="hibernate.transaction.factory_class">    
	  org.hibernate.transaction.JDBCTransactionFactory
      </property>    
</session-factory>
</hibernate-configuration>

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


我们来看下面这段代码:

Java代码
Transaction tx=null;
try{
session = sessionFactory.openSession();    
Transaction tx = session.beginTransaction();    //开启事务
//执行持久化操作
……    
tx.commit();   //操作正常,提交事务
}catch(RuntimeException e){
 if(tx!=null){
 tx.rollback();//操作过程中有一场,回滚事务
 throw e;//处理异常
}
}finally{
 session.close()
} 
  从JDBC层面而言,上面的代码实际上对应着: 
Java代码
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();    
2)  Hibernate中使用JTA事务
具体配置为
<?xml version="1.0" encoding="utf-8" ?>
<!DOCTYPE hibernate-configuration PUBLIC
	"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
	"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
	  <property name="hibernate.transaction.factory_class">    
	  org.hibernate.transaction.JTATransactionFactory
      </property>    
</session-factory>
</hibernate-configuration>
 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();    
……    
}    
}   
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(){
try{
UserTransaction tx = (UserTransaction)new InitialContext().lookup(“java:comp/UserTransaction”);
tx.begin();//开启JTA事务
ClassA.save(user);
ClassB.save(order);
tx.commit();//操作正常,提交JTA事务
}catch(RuntimeException e){
tx.rollback();//操作异常,回滚JTA事务
throw e;//异常处理
}finally{
}
}
}
    这里有两个类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();    
……    
}    
}   
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();    
……    
}    
}   
  实际上,这是利用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容器自动提交。
  
分享到:
评论

相关推荐

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

    - **配置**: 开发者需要在pom.xml或build.gradle文件中添加`hibernate-jpa-2.1-api-1.0.0.final.jar`依赖,然后在应用配置中指定Hibernate作为JPA提供商。 - **实体类**: 使用@Entity注解标记Java类为JPA实体,并...

    Hibernate稳定版(hibernate-release-5.3.23.Final.zip)

    Hibernate稳定版(hibernate-release-5.3.23.Final.zip),Hibernate ORM 是一个为应用程序、库和框架提供对象/关系映射 (ORM) 支持的库。它还提供了 JPA 规范的实现,这是 ORM 的标准 Java 规范。

    hibernate-core-5.0.11.Final.jar

    本文将重点探讨`hibernate-core-5.0.11.Final.jar`的核心源码,帮助读者深入理解其内部机制,并提供实用的应用指导。 一、Hibernate Core概述 Hibernate Core是Hibernate框架的基础部分,它包含了ORM的核心功能,...

    hibernate-jpa-2.0-api-1.0.1.Final.jar

    hibernate-jpa-2.0-api-1.0.1.Final.jar

    hibernate-jpa-2.0-api-1.0.1.Final-sources.jar

    hibernate-jpa-2.0-api-1.0.1.Final-sources.jar hibernate jpa 源代码

    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-commons-annotations-4.0.1.Final.jar

    hibernate-commons-annotations-4.0.1.Final.jar

    hibernate-cglib-repack-2.1_3.jar

    《深入理解Hibernate-CGLIB-2.1_3在Java开发中的应用》 在Java开发领域,Hibernate作为一款强大的对象关系映射(ORM)框架,极大地简化了数据库操作。而CGLIB则是一个代码生成库,它允许开发者在运行时动态创建子类...

    hibernate-release-4.2.4.Final.zip

    1. **Hibernate框架**:Hibernate是一个强大的ORM框架,它在Java应用中起到了桥梁作用,连接应用程序代码和关系数据库。通过Hibernate,开发者无需直接编写SQL语句,而是使用Java对象来操作数据,从而降低了数据库...

    hibernate-extensions-2.1.3.zip ,middlegen for hibernate

    《Hibernate扩展与Middlegen在Hibernate中的应用》 Hibernate是一个流行的Java对象关系映射(ORM)框架,它简化了数据库操作,将Java对象与数据库表之间的映射过程自动化。而`hibernate-extensions-2.1.3.zip`则...

    hibernate-release-4.2.4核心jar包

    `hibernate-jpa-2.0-api-1.0.1.Final.jar`包含Java Persistence API(JPA)的规范,这是Java标准版中的一个部分,定义了如何在Java应用程序中进行对象持久化。Hibernate作为JPA的实现之一,依赖这个库来提供JPA的...

    Hibernate稳定版(hibernate-release-5.6.2.Final.zip)

    Hibernate稳定版(hibernate-release-5.6.2.Final.zip),Hibernate ORM 是一个为应用程序、库和框架提供对象/关系映射 (ORM) 支持的库。它还提供了 JPA 规范的实现,这是 ORM 的标准 Java 规范。

    hibernate-release-4.3.1.Final.zip

    标题中的"hibernate-release-4.3.1.Final.zip"表明这是一个关于Hibernate ORM框架的版本4.3.1的发布包。Hibernate是Java开发中广泛使用的对象关系映射(ORM)工具,它允许开发者用面向对象的方式来操作数据库,从而...

    Spring与Hibernate集成---声明式事务

    总结来说,Spring与Hibernate的集成使我们可以轻松地在应用程序中管理数据访问和事务,同时保持代码的整洁和模块化。声明式事务管理让开发者专注于业务逻辑,而无需关心事务控制的细节,提高了开发效率和代码质量。...

    Hibernate 离线的配置方法(hibernate-mapping-3.0.dtd)

    因为Hibernate在读出hbm.xml文件时需要通过网络读取到hibernate-mapping-3.0.dtd 文件。 如果没有网络不能正常工作。 所以提供上述文件。 以及hibernate-mapping-3.0.dtd,hibernate-configuration-3.0.dtd提供下载...

    hibernate源码release-4.1.4.Final版

    Hibernate支持JTA和JDBC两种事务管理方式。在Java EE环境下,推荐使用JTA进行分布式事务控制;而在Java SE环境中,通常采用JDBC事务管理。 六、持久化操作 Hibernate提供了save()、update()、merge()和delete()方法...

    hibernate-mapping-3.0.dtd

    hibernate-mapping-3.0.dtd 最新从官网下载,无损无修改

    Hibernate稳定版(hibernate-release-5.2.18.Final.zip)

    Hibernate稳定版(hibernate-release-5.2.18.Final.zip),Hibernate ORM 是一个为应用程序、库和框架提供对象/关系映射 (ORM) 支持的库。它还提供了 JPA 规范的实现,这是 ORM 的标准 Java 规范。

    hibernate-release-5.0.7 全包

    本次我们将详细探讨hibernate-release-5.0.7全包的内容及其在实际开发中的应用。 该压缩包包含了Hibernate 5.0.7版本的核心组件以及其依赖库,用户可以直接将其拷贝到WEB-INF下的lib目录,无需额外配置,即可在Java...

    最新 hibernate-release-4.2.13.Final.tgz

    在“hibernate-release-4.2.13.Final”目录下,你会发现一系列的核心jar包,如hibernate-core.jar,它包含了Hibernate的主要功能,包括实体管理、查询API、事务处理等。此外,还有hibernate-entitymanager.jar,专注...

Global site tag (gtag.js) - Google Analytics