Spring事务@Transaction失效原因

发布时间 2023-12-26 18:15:31作者: 乐子不痞

Spring事务@Transaction不生效的原因

1、数据库引擎不支持事务

某些数据库引擎不支持事务,如果你使用这些引擎,则不能正确地使用@Transactional注解。

2、事务管理器配置问题

如果 Spring 配置文件中没有启用事务注解配置,或者配置不正确,事务不会生效。

或者开启了多个事务管理器,而事务注解没有指定使用哪个事务管理器,事务注解可能无法生效。确保事务注解中指定了正确的事务管理器。

3、没有被 Spring IOC管理

使用@Transactional进行事务管理时,Spring需要通过代理对象来管理事务。如果你的类没有被代理,则事务注解将无法生效。

// @Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void updateOrder(Order order) {
        // update order
    }
}

如果此时把 @Service 注解注释掉,这个类就不会被加载成一个 Bean,那这个类就不会被 Spring 管理了,事务自然就失效了。

4、非public方法

以下来自 Spring 官方文档:

When using proxies, you should apply the @Transactional annotation only to methods with public visibility. If you do annotate protected, private or package-visible methods with the @Transactional annotation, no error is raised, but the annotated method does not exhibit the configured transactional settings. Consider the use of AspectJ (see below) if you need to annotate non-public methods.

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

5、内部方法调用(常见)

事务的生效是基于AOP代理的,如果在同一个类中的一个方法内调用同类中的另一个方法,事务注解可能不会起作用,因为代理机制不会被激活。

来看两个示例:

@Service
public class OrderServiceImpl implements OrderService {

    public void update(Order order) {
        updateOrder(order);
    }

    @Transactional
    public void updateOrder(Order order) {
        // update order
    }

}

update方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法,updateOrder 方法上的事务会生效吗?(答案:不生效)

再来看下面这个例子:

@Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void update(Order order) {
        updateOrder(order);
    }

    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void updateOrder(Order order) {
        // update order
    }

}

这次在 update 方法上加了 @Transactional,updateOrder 加了 REQUIRES_NEW 新开启一个事务,那么新开的事务管用么?

这两个例子的答案是:不管用!

解决方法:

1、通过将该方法移动到一个单独的类中,并通过依赖注入调用来解决这个问题。

2、在的类中注入自己,用注入的对象再调用另外一个方法。

6、异常被捕获(常见)

如果在事务方法中捕获了异常并处理了它,但没有将其重新抛出,那么事务将不会回滚。

例子:

@Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void updateOrder(Order order) {
        try {
            // update order
        } catch {

        }
    }
}

7、非受检异常(常见)

Spring默认只会在遇到非受检异常(继承自RuntimeException)时回滚事务。如果你的事务方法抛出了受检异常(继承自Exception),则事务可能不会回滚。

可以通过在@Transactional注解中指定rollbackFor属性来指定回滚的异常类型。

@Service
public class OrderServiceImpl implements OrderService {

    @Transactional
    public void updateOrder(Order order) {
        try {
            // update order
        } catch {
            throw new Exception("更新错误");
        }
    }
}

这样事务也是不生效的,因为默认回滚的是:RuntimeException

如果你想触发其他异常的回滚,需要在注解上配置一下,如:

@Transactional(rollbackFor = Exception.class)
 public void updateOrder(Order order) {
   try {
     // update order
   } catch {
     throw new Exception("更新错误");
   }
 }

这些是一些可能导致Spring事务@Transactional不生效的常见原因。但要注意,每个问题都需要具体分析和调试,以确定根本原因。如果问题仍然存在,请仔细检查你的代码、配置和环境,并查阅相关文档和资料来解决问题。