1.数据库不支持事务

spring事务生效的前提是要所连的数据库要支持事务。如果底层的数据都不支持事务,那spring事务肯定就会失效,比如MySQL数据库使用MyISAM存储引擎,则Spring的事务就会失效。
2.事务方法未被spring管理
如果事务方法所在的类没有加载到Spring IOC容器中,也就是说,事务方法所在的类没有被Spring管理,则Spring事务会失效。例如一下情况:

  1. public class UserService {
  2. @Autowired
  3. private UserDao userDao;
  4. @Transactional(propagation = Propagation.REQUIRES_NEW)
  5. public void updateUserPassById(String pass, Long id){
  6. userDao.updateUserPassById(pass, id);
  7. }
  8. }

该UserService类中没有使用service注解,没有被spring管理。那updateUserPassById()方法的spring事务也就会失效。

3.方法没有被public修饰

spring事务默认生效的方法权限都必须为public
之所以会失效是因为在Spring AOP 代理时,TransactionInterceptor (事务拦截器)在目标方法执行前后进行拦截,Spring代理工厂在启动时,会扫描所有类和方法,并会检查目标方法的修饰符是否为 public,不是 public则不会获取@Transactional 的属性配置信息。
避坑:protected、private 修饰的方法上使用 @Transactional 注解,虽然事务无效,但不会有任何报错,这是我们很容犯错的一点。

  1. @Service
  2. public class UserService {
  3. @Autowired
  4. private UserDao userDao;
  5. @Transactional(propagation = Propagation.REQUIRES_NEW)
  6. public void updateUserPassById(String pass, Long id){
  7. userDao.updateUserPassById(pass, id);
  8. }
  9. }

4.同一个类中方法调用

如果同一个类中的两个方法分别为A和B,方法A上没有添加事务注解,方法B上添加了@Transactional事务注解,方法A调用方法B,则方法B的事务会失效。

5.没有配置事务管理器

如果在项目中没有配置Spring的事务管理器,即使使用了Spring的事务管理功能,Spring的事务也不会生效。
例如,没有在项目的配置类中配置如下代码。

  1. @Bean
  2. public PlatformTransactionManager transactionManager(DataSource dataSource) {
  3. return new DataSourceTransactionManager(dataSource);
  4. }

6.方法的事务传播类型不支持事务

如果内部方法的事务传播类型为不支持事务的传播类型,则内部方法的事务在Spring中会失效。
@Transactional(propagation = Propagation.NOT_SUPPORTED)
方法的事务传播类型为NOT_SUPPORTED,不支持事务,方法在spring事务会失效。

7.不正确的捕获异常

updateProductStockCountById()方法中使用try-catch代码块捕获了异常,即使updateProductStockCountById()方法内部会抛出异常,但也会被catch代码块捕获到,spring事务只有捕捉到了业务抛出去的异常,才能进行后续的处理,如果业务自己捕获了异常,则事务无法感知。spring事务也就会失效。
例如:catch后并没有抛出异常,就不会捕获到,导致事务失败。

8.错误标注异常类型

  1. @Service
  2. public class UserService {
  3. @Autowired
  4. private UserDao userDao;
  5. @Transactional(propagation = Propagation.REQUIRES_NEW)
  6. public void updateUserPassById(String pass, Long id){
  7. try {
  8. userDao.updateUserPassById(pass, id);
  9. } catch (Exception e) {
  10. logger.error("更新密码失败:", e.getMesaage());
  11. throw new Exception("更新密码失败");
  12. }
  13. }
  14. }

为何会失效呢?这是因为Spring中对于默认回滚的事务异常类型为RuntimeException,上述代码抛出的是Exception异常。
默认情况下,Spring事务中无法捕获到Exception异常,所以此时updateProductStockCountById()方法事务的回滚会失效。
Spring默认抛出了未检查unchecked异常(继承自 RuntimeException 的异常)或者 Error才回滚事务;其他异常不会触发回滚事务。如果在事务中抛出其他类型的异常,但却期望 Spring 能够回滚事务,就需要指定 rollbackFor属性。
可以手动标注事务异常类型,类似如下的操作:

  1. @Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)

注意:Spring事务注解@Transactional中的rollbackFor属性可以指定 Throwable 异常类及其子类。