首先来谈谈数据库事务 ,数据库事务有四种特性 原子性  一致性   隔离性  持久性。

 原子性:事务在提交时的任何失败将导致整个事务的失败

 一致性:事务的失败 所有被该事务影响的数据必须与事务之前保持一致

隔离性:事务在执行过程中对数据的修改在事务提交之前对其它事务不可见。

持久性:事务提交后对数据的影响是持久的。


2.既然事务是从数据库而来的那java事务是什么 它们之间有啥关系?

   一个应用程序对数据库的增删改查是通过jdbc来实现的 ,所以数据库事务也相应的转到java上来


3 java的事务类型分三种。jdbc事务 ,JTA事务 ,容器事务

  jdbc事务是connection对象控制的 ,connection支持自动提交和手动提交两种方式提交事务(具体翻阅jdbc文档) 。你可以把多条sql放入一个jdbc执行。 jdbc事务有个缺点就是不支持分布式事务,只局限与一个数据库 不能跨越多个数据库。

 

    2、JTA(Java Transaction API)事务

  JTA是一种高层的,与实现无关的,与协议无关的API,应用程序和应用服务器可以使用JTA来访问事务。

分布式事务处理——在两个或多个网络计算机资源上访问并且更新数据,这些数据可以分布在多个数据库上。JDBC驱动程序的JTA支持极大地增强了数据访问能力。

  如果计划用 JTA 界定事务,那么就需要有一个实现 javax.sql.XADataSource 、 javax.sql.XAConnection 和 javax.sql.XAResource 接口的 JDBC 驱动程序。一个实现了这些接口的驱动程序将可以参与 JTA 事务。一个 XADataSource 对象就是一个 XAConnection 对象的工厂。 XAConnections 是参与 JTA 事务的 JDBC 连接。

  您将需要用应用服务器的管理工具设置 XADataSource。(从应用服务器和 JDBC 驱动程序的文档中可以了解到相关的指导)

JNDI

XA 连接参与了 JTA 事务。这意味着 XA 连接不支持 JDBC 的自动提交功能。同时,应用程序一定不要对 XA 连接调用 java.sql.Connection.commit() 或者 java.sql.Connection.rollback() .

    相反,应用程序应该使用 UserTransaction.begin()、 UserTransaction.commit() 和 UserTransaction.rollback() .

    3、容器事务

  容器事务主要是J2EE应用服务器提供的,容器事务大多是基于JTA完成,这是一个基于JNDI的,相当复杂的API实现。相对编码实现JTA事务管理,我们可以通过EJB容器提供的容器事务管理机制(CMT)完成同一个功能,这项功能由J2EE应用服务器提供。这使得我们可以简单的指定将哪个方法加入事务,一旦指定,容器将负责事务管理任务。这是我们土建的解决方式,因为通过这种方式我们可以将事务代码排除在逻辑编码之外,同时将所有困难交给J2EE容器去解决。使用EJB CMT的另外一个好处就是程序员无需关心JTA API的编码,不过,理论上我们必须使用EJB.


    四、三种Java事务差异

    1、JDBC事务控制的局限性在一个数据库连接内,但是其使用简单。

    2、JTA事务的功能强大,事务可以跨越多个数据库或多个DAO,使用也比较复杂。

    3、容器事务,主要指的是J2EE应用服务器提供的事务管理,局限于EJB应用使用。


    五、总结

  Java事务控制是构建J2EE应用不可缺少的一部分,合理选择应用何种事务对整个应用系统来说至关重要。一般说来,在单个JDBC 连接连接的情况下可以选择JDBC事务,在跨多个连接或者数据库情况下,需要选择使用JTA事务,如果用到了EJB,则可以考虑使用EJB容器事务。


   六、JTA原理

Transaction(事务) 分两种

Local Transaction 和 Global Transaction 
涉及到一个Connection的Commit,称为Local Transaction 
涉及到多个Connection的Commit,称为Global Transaction

Local Transaction用JDBC事务实现是没有问题,然而Global Transaction的实现就无法保证了,所以不得不了解下JTA的神奇,但其实这样的事务也是存在问题的,下面将指出

 

两阶段提交 -- prepare 和 commit. 
假设有两个Connection, con1, con2, 大体的过程如下 



    1. con1 = XAResouce1.getConnection...       
    2. con2 = XAResouce2.getConnection...       
    3.       
    4. con1 do some thing.       
    5. con2 do some thing.       
    6. after they finish.       
    7.       
    8. pre1 = XAResouce1.prepare();       
    9. pre2 = XAResouce2.prepare();       
    10.       
    11. if( both pre1 and pre2 are OK){       
    12. 2 commit       
    13. }else {       
    14. 2 rollback       
    15. }


    在XAResouce1 and 2 commit的时候, 
    可能XAResouce1 commit() 成功了,XAResouce2 commit()失败了。 
    这时候,会抛出一个 “启发式异常”。程序可以处理这个异常。比如,XAResouce.recover()之类。 
    但一般情况下,还真没别的办法,需要数据管理员根据数据操作日志 undo所有的操作,或者恢复数据备份。 
    有的数据库在进行数据操作的时候,会生成一个“反操作”日志。比如,insert 对 delete, 等。 

     

    Global Transaction 需要XA接口(包括在JTA里面)的支持。

    import javax.sql.XAConnection; 
    import javax.transaction.xa.Xid; 
    import javax.transaction.xa.XAResource; 
    import javax.transaction.xa.XAException; 
    import javax.transaction.Transaction; 
    import javax.transaction.TransactionManager;

    其中的 
    javax.sql.XAConnection; 
    javax.transaction.xa.Xid; 
    javax.transaction.xa.XAResource;

    这些XA接口的实现,需要数据库的JDBC提供。 
    数据库本身要支持XA。数据库的JDBC也要提供XA的实现。

     

    TransactionManager的实现能够处理多个XAResouce(一个XAResouce list)的情况。 
    比如Tyrex。或JBoss等EJB Server的Transaction实现代码