@Transaction使用&@事务失效

Transaction是Spring提供的关于事务的支持,Spring事务管理分为编码式和声明式两种方式。声明式事务有两种方式,一种基于配置xml文件。另一种基于@Transactional注解的方式。

  1. @Transactional注解管理事务的实现步骤
  1. 在xml当中配置文件中添加事务配置信息。




  1. 将@Transactional注解添加到合适的方法上。

注: @Transactional注解可以添加到类级别上。当@Transactional注解放在类级别时,表示所有该类的公共方法都配置相同的事务属性信息。当类和方法都配置了@Transactional注解,应用程序以方法级别的事务属性来管理事务。也就是方法上的事务注解会覆盖掉类的事务注解配置

Spring的注解方式的事务实现机制

在应用系统调用声明@Transactional的目标方法时间,Spring Framework默认使用AOP代理,在代码运行时生成一个代理对象根据@Transactional的属性配置信息,这个代理对象决定声明@Transactional的目标是否由拦截器TransactionInterceptor来使用拦截。在拦截器拦截时,会在目标方法执行之前创建并且加入事务,执行目标逻辑,最后根据执行情况,利用AbstractPlatformTransactionManager操作数据源DataSource提交或者回滚

事务执行机制

SpringAOP 代理有CGLIB 和JDK代理两种方式。 CGLIB 调用的是inteceptor,JDK调用的是invoke方法。

注解方式的事务使用注意事项

  1. 正确的设置@Transactional的propagation属性,一下三种设置,事务不会发生回滚
1.  TransactionalDefinition.PROPAGATION_SUPPORTS:如果当前存在事务则加入该事务,如果没有事务,则以非事务方式继续进行
    
    
2.  TransactionalDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起,
    
    
3.  TransactionalDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常
  1. 正确的设置rollbackFor属性
1.  如果事务中抛出了未检查异常,或者ERROR,则Spring将会回滚事务;除此之外,Spring不会回滚事务。如果在事务中抛出其他类型的异常,并期望Spring能够回滚事务,可以指定rollbackFor。
  1. @Transactional只能应用到public方法才有效
1.  只有@Transactional注解应用到 public方法。因为AOP代理时在TransactionIntectptor在目标方法执行之前,代理对象的invoke或者inteceptor会间接调用AbstractFallbackTransactionAttributeSource中的computeTransactionAttribute。这个方法会判断目标方法是不是public。如果不是public,则不会获取事务的属性配置信息。
  1. 避免AOP自调用问题
1.  在SpringAop的代理下,只有目标方法由外部调用,目标方法才由Spring生成的代理对象来管理。若同一类当中没有@Transactional注解的方法内部调用@Transactional方法,有@Transactional注解的方法则会被忽略。

事务失效的原因

  1. 数据库引擎不支持事务。 以MySql为例,MyISAM引擎是不支持事务操作的,InnoDB才是支持事务的引擎,一般要支持事务都采用InnoDB。
  1. 没有被Spring管理
//@Service  当@Service注解注释掉,Spring不会管理该类,也不会变成一个bean,事务失效
 public class OrderServiceImpl implements OrderService {
 @Transactional
 public void updateOrder(Order order) {
 // update order
 }
}
  1. 方法不是public的

@Transactional只能用于public的方法上,否则事务会失效,如果非要用在非public上可以开启Aspectj代理模式

  1. 自身调用问题 在一个类当中,两个方法都采用了@Transactional注解的情况下,类的内部发生自调用,没有经过Spring代理类,事务也不会生效。默认只有在外部调用事务才会生效。
  1. 数据源没有配置事务管理器
public PlatformTransactionManager transactionManager(DataSource dataSource) {
 return new DataSourceTransactionManager(dataSource);
}

6.不支持事务 Propagation.NOT_SUPPORTED: 表示不以事务运行

  1. 异常类型错误 事务回滚的默认类型是RuntimeException,如果想触发其他异常,需要在注解上配置@Transactional(rollbackFor=Exception.class)配置的异常类只有Throwable异常类及其子类
  1. 未抛出异常 当try catch之后 为对异常做抛出动作,事务无法回滚。

你可能感兴趣的:(@Transaction使用&@事务失效)