Spring 事务失效的 8 大场景,看看你都遇到过几个

本文总结了8种导致Spring事务失效的场景,包括数据库引擎不支持事务、类未被Spring管理、方法非public、自身调用问题等,并针对每种情况提供了分析及解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

用 Spring 的 @Transactional 注解控制事务有哪些不生效的场景?

不知道小伙伴们有没有这样的经历,在自己开心的编写业务代码时候,突然某一个方法里的事务好像失效了。然后 debug 跟踪代码时发现,自己第一步的 insert 或者 update 的数据在语句执行完毕后,数据库中并没有立即出现更改或保存完的新数据。所以一度怀疑spring 的事务失效了。那么这篇文章就来总结一下,大家给大家造成 “spring事务失效”错觉的 几个常见场景,然后对症下药。

Let's GO!!!

以本人的经历中遇到的问题,大概分有以下几个场景:

  1. 数据库引擎是否支持事务(Mysql 的 MyIsam引擎不支持事务);

  2. 注解所在的类是否被加载为 Bean(是否被spring 管理);

  3. 注解所在的方法是否为 public 修饰的;

  4. 是否存在自身调用的问题;

  5. 所用数据源是否加载了事务管理器;

  6. @Transactional的扩展配置propagation是否正确。

下面展开分析每一个场景:

  • 数据库引擎不支持事务

这里以 MySQL 为例,其 MyISAM 引擎是不支持事务操作的,InnoDB 才是支持事务的引擎,一般要支持事务都会使用 InnoDB。

根据 MySQL 的官方文档:

https://dev.mysql.com/doc/refman/5.5/en/storage-engine-setting.html

从 MySQL 5.5.5 开始的默认存储引擎是:InnoDB,之前默认的都是:MyISAM,所以这点要值得注意,底层引擎不支持事务再怎么搞都是白搭。

  • 没有被 Spring 管理

如下面例子所示:

// @Service
public class OrderServiceImpl implements OrderService {

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

}

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

  •  方法不是 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 代理模式。

  • 自身调用问题

来看两个示例:

//示例1

@Service
public class OrderServiceImpl implements OrderService {

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

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

}
//示例2

@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
    }

}

示例1 中,update方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法,updateOrder 方法上的事务管用吗?

示例2 中,update方法上面加了 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法,updateOrder 方法上的事务管用吗?

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

因为它们发生了自身调用,就调该类自己的方法,而没有经过 Spring 的代理类,默认只有在外部调用事务才会生效,这也是老生常谈的经典问题了。

这个的解决方案之一就是在的类中注入自己,用注入的对象再调用另外一个方法,这个不太优雅,另外一个可行的方案可以参考《Spring 如何在一个事务中开启另一个事务?》这篇文章。

  • 数据源没有配置事务管理器

如下代码所示,当前数据源若没有配置事务管理器,那也是白搭!

@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
    return new DataSourceTransactionManager(dataSource);
}
  • @Transactional的扩展配置不支持事务

Propagation.NOT_SUPPORTED: 表示不以事务运行,当前若存在事务则挂起。这表示不支持以事务的方式运行,所以即使事务生效也是白搭!

@Service
public class OrderServiceImpl implements OrderService {

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

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

}
  • 异常被吃了

这个也是出现比较多的场景:把异常吃了,然后又不抛出来,事务也不会回滚!

@Service
public class OrderServiceImpl implements OrderService {

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

        }
    }

}
  • 异常类型错误

接上面的例子,再抛出一个异常

@Service
public class OrderServiceImpl implements OrderService {

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

}

这样事务也是不生效的,因为默认回滚的是:RuntimeException,如果你想触发其他异常的回滚,需要在注解上配置一下,如:

@Transactional(rollbackFor = Exception.class)

这个配置仅限于 Throwable 异常类及其子类。

总结:本文总结了 8 种事务失效的场景,其实发生最多就是自身调用、异常被吃、异常抛出类型不对这 3 个了,像文章开头说的那样,本文不一定总结得全,只是总结常见的事务失效的场景,如果你还知道其他场景也欢迎留言分享。

<think>嗯,用户这次问的是如何监控Spring框架中的事务行为。从引用的内容来看,用户应该已经对Spring事务的基本机制有所了解,比如@Transactional注解的工作原理、回滚规则等。用户可能是正在开发或维护一个使用Spring事务管理的应用,遇到了事务相关的问题需要排查。用户引用的资料提到了几个关键点:PlatformTransactionManager管理事务生命周期、回滚规则、TransactionSynchronizationManager工具类,这些都对监控很有帮助。用户应该需要知道具体怎么操作,而不仅仅是理论。我想到几个监控方向:一是通过日志输出事务活动,二是用编程方式主动检查事务状态,三是通过性能监控工具观察事务指标,四是利用APM工具自动追踪。这些方法可以形成互补。日志监控比较直接,但可能需要调整日志级别,特别是对TransactionSynchronizationManager和AbstractPlatformTransactionManager的记录。编程检查更主动,比如用isActualTransactionActive()判断事务是否存活,用holder保存关键事务信息等。性能监控侧重指标,可以观察事务数量、持续时间、异常率等。用户引用的[^3]特别提到了TransactionSynchronizationManager,这个工具类确实很实用,应该重点介绍。还有[^5]提到的异常处理也很重要,监控时要注意不同异常类型的事务回滚情况。用户可能还需要一些实例代码,比如怎么在TransactionSynchronization中记录事务时间戳。考虑到用户是开发者,给些代码片段会更直观。最后,用户提到"尽可能参考上一次的问题",但目前看不到上下文,可能是初次交流?不过根据提供的引用内容,重点围绕事务监控展开回答应该就符合需求。</think> 监控Spring事务行为是保证应用数据一致性的关键手段,以下是主要方法与工具: --- ### **一、编程式监控核心方法** 1. **`TransactionSynchronizationManager`工具类** - **判断事务活跃状态** ```java boolean isActive = TransactionSynchronizationManager.isActualTransactionActive(); ``` *用途*:检查事务是否生效(如验证`@Transactional`是否失效)[^3] - **获取事务属性** ```java String txName = TransactionSynchronizationManager.getCurrentTransactionName(); IsolationLevel isolation = TransactionSynchronizationManager.getCurrentTransactionIsolationLevel(); ``` 2. **事务同步回调(`TransactionSynchronization`)** 注册回调监听事务生命周期: ```java TransactionSynchronizationManager.registerSynchronization( new TransactionSynchronization() { @Override public void afterCommit() { // 事务提交后触发 } @Override public void afterCompletion(int status) { // 事务完成时触发(包含回滚) } } ); ``` *典型场景*:审计日志、异步操作提交后执行[^1] --- ### **二、日志监控配置** 1. **开启Spring事务日志** 在`application.properties`中添加: ``` logging.level.org.springframework.transaction.interceptor=TRACE logging.level.org.springframework.jdbc.datasource.DataSourceTransactionManager=DEBUG ``` *输出内容*:事务开启/提交/回滚时间、传播行为、隔离级别。 2. **事务事件日志示例** ```log 2023-10-01 DEBUG: Creating new transaction with name [com.example.Service.updateData] Propagation=REQUIRED, Isolation=DEFAULT 2023-10-01 DEBUG: Committing JDBC transaction on Connection [...] 2023-10-01 DEBUG: Rolling back JDBC transaction due to [NullPointerException] ``` --- ### **三、APM工具监控(生产环境推荐)** | 工具 | 监控能力 | |---------------|--------------------------------------------------------------------------| | **SkyWalking** | 可视化事务链路、耗时统计、异常回滚标记 | | **Micrometer** | 通过`@Timed("transaction")`注解采集事务延迟、成功率 | | **Prometheus** | 收集事务指标:`spring_transactions_active{status="committed"}` | | **Dynatrace** | 自动检测Spring事务边界,关联数据库调用耗时 | --- ### **四、事务状态诊断技巧** 1. **异常回滚验证** ```java @Transactional public void update() { try { jdbcTemplate.execute("UPDATE ..."); // 业务操作 throw new IOException(); // 受检异常 } catch (Exception e) { // 需手动设置回滚(默认不回滚受检异常) TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); } } ``` *注意*:默认仅回滚`RuntimeException`和`Error`[^5] 2. **连接持有监控** 通过`DataSource`代理记录事务持有连接时长,避免长时间未提交导致连接池耗尽。 --- ### **
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值