java开发中对事务的学习总结
数据库中对事务的定义,我相信大多数人都已经有一个大致的了解了:四个原则(ACID)。
这里我主要总结一下,我遇到的一个项目中对事务的管理是如何去实现的,以及我在学习的过程中的一些知识总结。
1. 事务的类型:
a)从数据库的角度来看,事务分为本地事务跟全局事务
1. 本地事务:普通事务,独立一个数据库,能保证在该数据库上操作的ACID;
2. 分布式事务:涉及两个或多个数据库源的事务,即跨越多台同类或异类数据库的事务(由每台数据库的本地事务组成的),分布式事务旨在保证这些本地事务的所有操作的ACID,使事务可以跨越多台数据库。
b)从Java的角度来看,又分为3种:JDBC事务、JTA(Java Transaction API)事务、容器事务。
1. JDBC事务:即数据库事务中的本地事务,通过connection对象控制管理。
2. JTA事务:JTA指Java事务API(Java Transaction API),是Java EE数据库事务规范, JTA只提供了事务管理接口,由应用程序服务器厂商(如WebSphere Application Server)提供实现,JTA事务比JDBC更强大,支持分布式事务。
3. 容器事务:容器事务主要是J2EE应用服务器提供的,容器事务大多是基于JTA完成,这是一个基于JNDI的,相当复杂的API实现。采用这种事务方式,相理论上我们必须使用EJB。
2. spring中事务的管理:
Spring并不直接管理事务,而是提供了多种事务管理器,他们将事务管理的职责委托给Hibernate或者JTA等持久化机制所提供的相关平台框架的事务来实现。 其核心接口的联系如下:
我遇到的这个项目就是采用 org.springframework.transaction.jta.JtaTransactionManager 事务管理器,配合 Atomikos 进行分布式事务的管理。
最后在代码中采用 @Transactional注解的方式声明事务。
<!-- transaction declare method 1 : annotation -->
<tx:annotation-driven transaction-manager="transactionManager" />
<!-- transaction declare method 2 : AOP -->
<!-- <tx:advice id="userTxAdvice" transaction-manager="transactionManager"> -->
<!-- <tx:attributes> -->
<!-- <tx:method name="txDelete*" propagation="REQUIRED" -->
<!-- read-only="false" rollback-for="java.lang.Exception" -->
<!-- no-rollback-for="java.lang.RuntimeException" /> -->
<!-- <tx:method name="txInsert*" propagation="REQUIRED" -->
<!-- read-only="false" rollback-for="java.lang.RuntimeException" /> -->
<!-- <tx:method name="txUpdate*" propagation="REQUIRED" -->
<!-- read-only="false" rollback-for="java.lang.Exception" /> -->
<!-- <tx:method name="txRunCustom*" propagation="REQUIRED" -->
<!-- read-only="false" rollback-for="java.lang.Exception" /> -->
<!-- </tx:attributes> -->
<!-- </tx:advice> -->
<!-- <aop:config> -->