最近在做事务相关的工作时总是有点模糊,需要搜索确定事务是否会生效,又不能每次都写个demo实验一下,所以把相关的知识点都总结一下。
事务不生效的场景
- 数据库引擎 mysql的MyISAM,是不支持事务操作的。需要改成InnoDB才能支持。
- 访问修饰符 public修饰的方法入口才能生效,否则springAop不切入,private 方法, final 方法 和 static 方法不能添加事务,加了也不生效。
- 异常问题 spring事务只对运行时异常进行回滚。
- 没有开启事务管理注解@EnableTransactionManagement |<tx:annotation-driven />
- 该类不是spring的代理类
- 业务逻辑和事务不在同一个线程里
方法调用的事务生效
- 无事务的方法调用另一个有事务的方法,事务不会起作用
- 有事务的方法调用另一个有事务的方法,事务起作用
- 有事务的方法调用另一个无事务的方法,事务起作用
- 无事务的方法调用另一个有事务的方法,使用自己的service来调用,或者使用代理类来调用,事务生效
事务的传播机制
- PROPAGATION_REQUIRED – 支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
- PROPAGATION_SUPPORTS – 支持当前事务,如果当前没有事务,就以非事务方式执行。
- PROPAGATION_MANDATORY – 支持当前事务,如果当前没有事务,就抛出异常。
- PROPAGATION_REQUIRES_NEW – 新建事务,如果当前存在事务,把当前事务挂起。
- PROPAGATION_NOT_SUPPORTED – 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
- ROPAGATION_NEVER – 以非事务方式执行,如果当前存在事务,则抛出异常。
- PROPAGATION_NESTED – 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。
- 前六个策略类似于EJB CMT,第七个(PROPAGATION_NESTED)是Spring所提供的一个特殊变量。它要求事务管理器或者使用JDBC 3.0 Savepoint API提供嵌套事务行为(如Spring的DataSourceTransactionManager)