Transaction是Spring提供的关于事务的支持,Spring事务管理分为编码式和声明式两种方式。声明式事务有两种方式,一种基于配置xml文件。另一种基于@Transactional注解的方式。
- @Transactional注解管理事务的实现步骤
- 在xml当中配置文件中添加事务配置信息。
<tx:annotation-driven />
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
- 将@Transactional注解添加到合适的方法上。
注: @Transactional注解可以添加到类级别上。当@Transactional注解放在类级别时,表示所有该类的公共方法都配置相同的事务属性信息。当类和方法都配置了@Transactional注解,应用程序以方法级别的事务属性来管理事务。也就是方法上的事务注解会覆盖掉类的事务注解配置
Spring的注解方式的事务实现机制
在应用系统调用声明@Transactional的目标方法时间,Spring Framework默认使用AOP代理,在代码运行时生成一个代理对象根据@Transactional的属性配置信息,这个代理对象决定声明@Transactional的目标是否由拦截器
TransactionInterceptor
来使用拦截。在拦截器拦截时,会在目标方法执行之前创建并且加入事务,执行目标逻辑,最后根据执行情况,利用AbstractPlatformTransactionManager
操作数据源DataSource提交或者回滚SpringAOP 代理有CGLIB 和JDK代理两种方式。 CGLIB 调用的是inteceptor,JDK调用的是invoke方法。
注解方式的事务使用注意事项
- 正确的设置@Transactional的propagation属性,一下三种设置,事务不会发生回滚
1. TransactionalDefinition.PROPAGATION_SUPPORTS:如果当前存在事务则加入该事务,如果没有事务,则以非事务方式继续进行 2. TransactionalDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起, 3. TransactionalDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常
- 正确的设置rollbackFor属性
1. 如果事务中抛出了未检查异常,或者ERROR,则Spring将会回滚事务;除此之外,Spring不会回滚事务。如果在事务中抛出其他类型的异常,并期望Spring能够回滚事务,可以指定rollbackFor。
- @Transactional只能应用到public方法才有效
1. 只有@Transactional注解应用到 public方法。因为AOP代理时在TransactionIntectptor在目标方法执行之前,代理对象的invoke或者inteceptor会间接调用AbstractFallbackTransactionAttributeSource中的computeTransactionAttribute。这个方法会判断目标方法是不是public。如果不是public,则不会获取事务的属性配置信息。
- 避免AOP自调用问题
1. 在SpringAop的代理下,只有目标方法由外部调用,目标方法才由Spring生成的代理对象来管理。若同一类当中没有@Transactional注解的方法内部调用@Transactional方法,有@Transactional注解的方法则会被忽略。
事务失效的原因
- 数据库引擎不支持事务。 以MySql为例,MyISAM引擎是不支持事务操作的,InnoDB才是支持事务的引擎,一般要支持事务都采用InnoDB。
- 没有被Spring管理
//@Service 当@Service注解注释掉,Spring不会管理该类,也不会变成一个bean,事务失效
public class OrderServiceImpl implements OrderService {
@Transactional
public void updateOrder(Order order) {
// update order
}
}
- 方法不是public的
@Transactional只能用于public的方法上,否则事务会失效,如果非要用在非public上可以开启Aspectj代理模式
- 自身调用问题 在一个类当中,两个方法都采用了@Transactional注解的情况下,类的内部发生自调用,没有经过Spring代理类,事务也不会生效。默认只有在外部调用事务才会生效。
- 数据源没有配置事务管理器
public PlatformTransactionManager transactionManager(DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
}
6.不支持事务 Propagation.NOT_SUPPORTED: 表示不以事务运行
- 异常类型错误 事务回滚的默认类型是RuntimeException,如果想触发其他异常,需要在注解上配置@Transactional(rollbackFor=Exception.class)配置的异常类只有Throwable异常类及其子类
- 未抛出异常 当try catch之后 为对异常做抛出动作,事务无法回滚。