spring与事务管理
就我接触到的事务,使用最多的事务管理器是JDBC事务管理器。现在就记录下在spring中是如何使用JDBC事务管理器
1)在spring中配置事务管理器
<bean id="jdbcTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"></property>
</bean>
为啥要为DataSourceTransactionManager类装配dataSource Bean?
这是因为DataSourceTransactionManager是调用java.sql.Connection来管理事务的。Connection是通过DataSource获取到的。通过调用Connection.commit()来提交事务;调用Connection.rollback()来回滚事务。所以需要装配DataSource这个Bean.
2)了解事务的五边形:传播行为、隔离级别、只读、事务超时、事务规则
声明事务的时候,需要指明方法是否要在事务环境中运行,事务的隔离级别等等,要先了解事务的五边形:
以下是事务的五边形整理:
传播行为 |
Propagation_mandatory |
表示方法必须在事务中运行,若事务不存在,则抛出一个异常 |
Propagation_nested |
若当前存在一个事务,则方法会在嵌套事务中运行,如果不存在,则与propagation_required一样 | |
Propagation_never |
方法不运行在事务环境中,若存在当前事务,则抛出异常 | |
Propagation_not_supports |
方法不应该运行在事务中,若存在当前事务,方法运行时,当前事务将被挂起 | |
Propagation_required |
方法必须运行在事务中,若没有事务,则会启动一个新的事务 | |
Propagation_required_new |
方法必须运行在它自己事务中,方法运行时,若存在当前事务,事务被挂起,同时会启动一个新的事务 | |
Propagation_supports |
方法不需要事务上下文,但存在当前事务,会在事务中运行 | |
隔离级别 |
Isolation_read_uncommitted |
允许读取未提交的数据,会导致脏读,不可重复读,幻读 |
Isolation_read_committed |
允许读取并发事务提交的数据。如:事务A在运行时,第一次读取到a,此时并发事务B更新了数据,然后事务A第二次读取到b,导致a与b不一样。因此可阻止脏读,可导致不可重复读,幻读 | |
Isolation_repeatable_read |
对同一字段的多次读取是一样的,除非数据是本事务修改。这样能够防止不可重复读出现;当事务A运行时,读取几行数据后,并发事务B此时插入了一些数据,此时事务A继续读取,这样A读取的数据多了一些原本不存在的记录。因此可阻止脏读,不可重复读,可导致幻读 | |
Isolation_serializable |
完全锁定事务相关表,可阻止脏读,不可重复读,幻读 | |
只读 |
read_only |
若对db只做读操作;可将事务声明为只读,这样db可进行只读优化;当事务启动时,若该事务声明为只读,会通知db实施只读优化;因此需要传播行为可能具有新启动事务才能有效 |
事务超时 |
超时时钟会在事务启动时开始,因此只有具备可能可动事务的传播行为才有效 | |
回滚规则 |
默认情况下,事务在检查型异常时是不会回滚,只有在运行期异常才会回滚;通过设置回滚规则,可声明事务在遇到特定的异常不回滚,即使是在运行期。 |
3)在xml中声明事务
① 使用<tx:advice> 声明一个事务性策略AOP通知
<tx:advice id="txAdvice" transaction-manager="jdbcTransactionManager">
<tx:attributes>
<tx:method name="save*" propagation="REQUIRED"/>
<tx:method name="*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</tx:advice>
以上通知表示:以save开头的方法必须运行在事务中。其他方法的运行不需要事务上下文存在,如果存在当前事务,则会在事务中运行。且声明为只读,这样当方法运行时启动一个新事务进行的是读操作,
可让db实施只读优化;
②现在只声明的是一个AOP通知,并没有指明哪些Bean是需要配置事务的。所以需要一个切点
<aop:config>
<aop:advisor advice-ref="txAdvice" pointcut="execusion(* com.test.demo.dao.impl.*.*(..))"/>
</aop:config>
4)使用注解驱动的事务
相比在xml中声明事务,使用注解驱动的事务更加简洁:
<tx:annotation-driven>会告诉spring在spring上下文中查找使用了@Transctional注解的Bean,不管这注解是在类级别上还是方法上。对于每个使用了@Transactional注解的Bean,<tx:annotation-driven>会自动为它添加事务通知。
public class UserDaoAnn implements UserDao {
@Transactional(propagation=Propagation.REQUIRED,readOnly=false)
public void addUser(Monkey Monkey) {
// TODO Auto-generated method stub
}
}
类UserDaoAnn,类级别上使用了@Transactional(propagation=Propagation.SUPPORTS,readOnly=true),表示方法不需要运行在事务上下文中,并为事务声明为只读
方法级别上addUser,使用了@Transactional(propagation=Propagation.REQUIRED,readOnly=false),表示方法必须运行在事务中