Spring 同方法自调用导致 @Transactional 失效

一、现场日志(还原案发)

Service 层代码:

java 复制代码
@Service
public class OrderService {
    
    public void create(OrderDTO dto) {
        // 参数校验 & 业务计算
        Order order = convert(dto);
        // ② 调用本类"事务"方法
        saveOrder(order);
    }
    
    @Transactional(rollbackFor = Exception.class)
    public void saveOrder(Order order) {
        orderMapper.insert(order);   // ①
        orderDetailMapper.batchInsert(order.getDetails()); // ③
        if (order.getDetails().size() > 10) {
            throw new IllegalArgumentException("明细过多");
        }
    }
}

单元测试

java 复制代码
@Test
public void shouldRollbackWhenTooManyDetails() {
    OrderDTO dto = buildBigOrder(); // 11 条明细
    Assertions.assertThrows(IllegalArgumentException.class,
        () -> orderService.create(dto));
    // 期望:订单表 0 条,明细表 0 条
    assertEquals(0, jdbcTemplate.queryForObject(
        "select count(*) from t_order", Long.class));
}

结果

  • 异常确实抛出 ✅

  • 数据库仍出现 1 条订单 + 11 条明细 ❌

  • 日志无 rollback 信息 ❌

二、根因剖析

1. Spring 事务 = 动态代理

  • 容器启动时为 OrderService 生成 TransactionProxy(CgLib)

  • 代理类重写 saveOrder(),伪代码:

    java 复制代码
    public void saveOrder(Order order) {
        TransactionStatus tx = PlatformTransactionManager.getTransaction(new DefaultTransactionDefinition());
        try {
            // 目标对象真实方法
            target.saveOrder(order);
            PlatformTransactionManager.commit(tx);
        } catch (Throwable e) {
            PlatformTransactionManager.rollback(tx);
            throw e;
        }
    }

2. 自调用 bypass 代理

  • create()目标对象内部方法this.saveOrder(order)this = 原生对象,不是代理

  • 因此事务切面永远不被触发,等同于普通本地方法调用

https://i.imgur.com/tx-spring-proxy.png

三、复现 Demo(最小可运行)

com.example.chapter1.tx.SelfInvokeDemoApplicationTests

关键断点:在 OrderMapper.insert 后查看 TransactionSynchronizationManager.getCurrentTransactionName()

  • 自调用场景返回 null(无事务)

  • 代理调用返回 saveOrder(存在事务)


四、正确修复方案

|-------------------|-------------------------------------------------------------------|-----------|------------|
| 方案 | 代码示例 | 优点 | 缺点 |
| 拆 Service(推荐) | OrderCommandService.saveOrder() | 完全解耦、事务生效 | 类增多 |
| 注入代理自调用 | @Autowired private OrderService self; + self.saveOrder(order) | 改动小 | 循环依赖风险 |
| AspectJ 编译织入 | <tx:annotation-driven mode="aspectj"/> | 无代理限制 | 构建复杂、需 ajc |
| 编程式事务 | TransactionTemplate.execute(status -> { ... }); | 灵活、可读写分离 | 模板代码多 |

拆 Service 示例

java 复制代码
@Service
public class OrderCommandService {
    @Transactional(rollbackFor = Exception.class)
    public void saveOrder(Order order) {
        orderMapper.insert(order);
        orderDetailMapper.batchInsert(order.getDetails());
        if (order.getDetails().size() > 10)
            throw new IllegalArgumentException("明细过多");
    }
}

@Service
@RequiredArgsConstructor
public class OrderService {
    private final OrderCommandService commandService;
    
    public void create(OrderDTO dto) {
        Order order = convert(dto);
        commandService.saveOrder(order); // 走代理
    }
}

单元测试再次执行

✅ 异常抛出后数据回滚,表记录 0 条


五、防御体系(防止再犯)

  1. ArchUnit 规则固化
java 复制代码
@ArchTest
static final ArchRule no_self_call_with_tx =
    noMethods().that().areAnnotatedWith(Transactional.class)
               .should().beCalledFromMethods().that().areDeclaredIn(
                   new DescribedPredicate<>("同一类") {
                       public boolean apply(JavaMethod input) {
                           return input.getOwner().equals(
                               input.getCallsOfSelf().get(0).getOwner());
                       }
                   });

测试失败即阻断 CI

  1. IDE 插件

    IntelliJ "Spring AOP" 检查已能实时警告自调用

  2. Code Review 清单

    • 本类方法是否含 @Transactional / @Cacheable / @Async

    • 若有,必须拆 Service 或注入代理

六、延伸阅读 & 工具
  • 调试:-Dorg.springframework.transaction.interceptor=TRACE 看事务边界

  • Arthas:watch com.example.OrderService saveOrder '{params,throwExp}' -e -x 3 确认是否进入 TransactionInterceptor

七、一句话总结
  • "只要还在同一个类里 `this.xxx()` 调用带 `@Transactional` 的方法,事务就 100% 失效;
  • 把方法搬到另一个 Bean,或者用代理/编程式事务,才能让 Spring 真正帮你管事务。"
相关推荐
一念一花一世界4 分钟前
Arbess从基础到实践(5) - 集成GitLab+SonarQube搭建Java项目自动化部署
java·gitlab·sonarqube·cicd·arbess
萧曵 丶4 分钟前
CompletableFuture 实际场景使用案例
java·多线程·并发编程·高级开发
_UMR_16 分钟前
多线程场景的学习3,使用CountDownLatch
java·开发语言
无限大.16 分钟前
验证码对抗史
java·开发语言·python
明月别枝惊鹊丶32 分钟前
【C++】GESP 三级手册
java·开发语言·c++
毕设源码-钟学长33 分钟前
【开题答辩全过程】以 公交线路查询系统为例,包含答辩的问题和答案
java
梵得儿SHI33 分钟前
SpringCloud - 核心组件精讲:Nacos 深度解析(服务注册 + 配置中心一站式实现)
java·spring boot·spring cloud·nacos·微服务架构的核心组件·服务注册发现与配置管理·nacos的核心原理与实战应用
不如打代码KK35 分钟前
Java SPI与Spring Boot SPI的区别
java·开发语言·spring boot
非凡的小笨鱼37 分钟前
利用arthas查看java服务里指定对象的大小
java·spring·arthas
代码or搬砖44 分钟前
自定义注解全面详解
java·开发语言