同一个类中,事务方法调用另一个事务方法

在同一个类中,事务方法调用另一个事务方法时,事务可能不会生效。这是因为 Spring 的事务管理是基于 AOP 代理实现的。当一个事务方法在同一个类中直接调用另一个事务方法时,内部调用不会通过代理,因此不会触发事务管理逻辑。

以下是一个具体示例,演示了这种情况:
经评论区大神提醒,两个@Transactional注解同时存在是可以生效的。当只有一个@Transactional时事务失效。

@Service
public class MyService {

    @Autowired
    private MyRepository myRepository;

    @Transactional
    public void methodA() {
        methodB();  // 内部调用,不会触发事务管理
        myRepository.save(new MyEntity());
    }

    @Transactional
    public void methodB() {
        // 事务逻辑
    }
}

在上述代码中,methodA 调用了 methodB,但由于这是一个内部方法调用,Spring AOP 代理不会拦截这个调用。因此,methodB 中的事务配置不会生效。

如何解决这个问题

  1. 使用 @Transactional 在外部类中调用
    将事务方法放在不同的类中,并通过外部类调用它们。

    @Service
    public class MyServiceA {
    
        @Autowired
        private MyServiceB myServiceB;
    
        @Transactional
        public void methodA() {
            myServiceB.methodB();  // 通过代理调用,事务将生效
            myRepository.save(new MyEntity());
        }
    }
    
    @Service
    public class MyServiceB {
    
        @Autowired
        private MyRepository myRepository;
    
        @Transactional
        public void methodB() {
            // 事务逻辑
        }
    }
    
  2. 使用代理注入自身
    使用 Spring 的 AopContext 获取当前代理对象,并通过代理对象调用事务方法。

    @Service
    public class MyService {
    
        @Autowired
        private MyRepository myRepository;
    
        @Transactional
        public void methodA() {
            ((MyService) AopContext.currentProxy()).methodB();  // 通过代理调用,事务将生效
            myRepository.save(new MyEntity());
        }
    
        @Transactional
        public void methodB() {
            // 事务逻辑
        }
    }
    
  3. 使用 ApplicationContext 获取代理对象
    在调用事务方法时,通过 ApplicationContext 获取代理对象。

    @Service
    public class MyService {
    
        @Autowired
        private MyRepository myRepository;
    
        @Autowired
        private ApplicationContext applicationContext;
    
        @Transactional
        public void methodA() {
            MyService proxy = applicationContext.getBean(MyService.class);
            proxy.methodB();  // 通过代理调用,事务将生效
            myRepository.save(new MyEntity());
        }
    
        @Transactional
        public void methodB() {
            // 事务逻辑
        }
    }
    

通过这些方法,可以确保在同一个类中调用事务方法时,事务配置依然生效。

### Java事务传播行为分析 在 Spring 框架中,事务传播行为定义了一个事务方法调用时,如何处理当前事务以及新事务的关系。当一个事务方法调用一个中的事务方法时,其行为取决于 `@Transactional` 注解所指定的传播属性。 #### 1. 同一代理对象内的方法调用 在同一中,如果一个方法调用一个带有 `@Transactional` 的方法,则由于 Spring 使用动态代理机制实现事务管理,默认情况下内部方法调用不会触发事务拦截器[^1]。这意味着即使目标方法声明了事务特性,实际执行过程中也不会应用这些事务规则。 #### 2. 不同之间的方法调用一个事务方法调用中的事务方法时,Spring AOP 动态代理能够正常工作,因此可以正确激活事务传播行为。以下是几种常见的传播模式及其作用: - **REQUIRED**: 这是最常用的传播级别。如果当前存在事务,则支持该事务;如果没有现有事务,则会启动一个新的事务[^3]。 - **REQUIRES_NEW**: 创建新的独立事务。无论外部是否有活动事务,此方法始终运行在一个全新的事务上下文中,并且原有事务会被挂起直到完成[^3]。 - **NESTED**: 嵌套事务允许子事务作为父事务的一部分而存在。它提供了回滚的能力而不影响整个大事务的成功状态[^3]。 - **SUPPORTS**, **NOT_SUPPORTED**, **MANDATORY**, 和 ** NEVER** : 这些选项分别表示不同的策略来应对已有或者不存在的情况下的操作需求. #### 示例代码展示跨事务交互 假设我们有两个服务 Bean:`MyFirstService` 和 `MySecondService`, 它们各自拥有自己的业务逻辑并标注相应的事务配置: ```java @Service public class MyFirstService { private final MySecondService mySecondService; public MyFirstService(MySecondService mySecondService){ this.mySecondService = mySecondService; } @Transactional public void startTransaction(){ try{ // Some DB operation here... mySecondService.performAnotherOperation(); }catch(Exception e){ throw new RuntimeException(e); } } } @Service public class MySecondService { @Transactional(propagation = Propagation.REQUIRES_NEW) public void performAnotherOperation(){ // Another set of DB operations that need their own transactional context. } } ``` 在这个场景下,`startTransaction()` 方法开启了第一个事务,在其中又调用了来自不同 bean (`mySecondService`) 的第二个事务方法 `performAnotherOperation()`. 根据设定好的 propagation 属性 (即 REQUIRES_NEW),后者将在完全分离的新事务里被执行[^4]。 ### 总结 对于跨越多个的服务层组件间相互调用涉及到了复杂的事务控制流程设计考量因素较多。开发者需清楚理解每种型的事务传播含义以便合理安排各部分功能模块间的协作关系从而达到预期效果。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值