当前位置: 首页 > news >正文

深入解析 Spring @Transactional 的事务开启机制

一、核心问题:无数据库操作时事务是否开启?

当方法被标记为 @Transactional 但内部没有数据库操作时,事务是否会被开启?
答案是:取决于事务管理器的实现和传播行为。以下分步骤解析其底层逻辑。


二、事务开启的底层流程

1. 代理对象的拦截机制

Spring 通过动态代理(JDK 动态代理或 CGLIB)为 @Transactional 方法生成代理对象。代理对象在方法调用时拦截请求,执行事务管理逻辑。

关键代码流程:
// 伪代码:代理对象的拦截逻辑
public Object intercept(Method method, Object[] args) {// 1. 检查方法是否标记为 @Transactionalif (isTransactional(method)) {// 2. 获取事务属性(传播行为、隔离级别等)TransactionAttribute txAttr = getTransactionAttribute(method);// 3. 通过事务管理器创建事务TransactionStatus status = transactionManager.getTransaction(txAttr);try {// 4. 执行原始方法Object result = method.invoke(target, args);// 5. 提交事务transactionManager.commit(status);return result;} catch (Exception ex) {// 6. 回滚事务transactionManager.rollback(status);throw ex;}} else {return method.invoke(target, args);}
}

2. 事务创建的触发条件

事务的创建由 transactionManager.getTransaction() 触发,而非数据库操作。
即使方法中没有数据库操作,只要满足以下条件,事务仍会被创建:

  • 传播行为要求开启新事务(如 REQUIRES_NEW)。
  • 当前没有事务存在(如默认的 REQUIRED 行为)。

三、事务管理器的具体行为

1. JDBC 事务管理器(DataSourceTransactionManager)

  • 行为:在 getTransaction() 时立即获取数据库连接,并开启事务。
  • 示例
    @Transactional
    public void emptyMethod() {// 无数据库操作
    }
    
    日志输出
    DEBUG o.s.j.d.DataSourceTransactionManager - Creating new transaction
    DEBUG o.s.j.d.DataSourceTransactionManager - Acquired Connection [conn1] for JDBC transaction
    DEBUG o.s.j.d.DataSourceTransactionManager - Committing JDBC transaction on Connection [conn1]
    
    即使没有数据库操作,连接仍会被获取,事务会被提交。

2. JPA 事务管理器(JpaTransactionManager)

  • 行为:延迟获取数据库连接,直到实际执行数据库操作。
  • 示例
    @Transactional
    public void emptyMethod() {// 无数据库操作
    }
    
    日志输出
    DEBUG o.s.orm.jpa.JpaTransactionManager - Creating new transaction
    DEBUG o.s.orm.jpa.JpaTransactionManager - Committing JPA transaction
    
    事务会被创建,但未实际获取数据库连接(无 Connection 日志)。

四、传播行为的影响

1. REQUIRED(默认)

  • 如果当前没有事务,开启新事务。
  • 即使方法无数据库操作,事务仍会被创建(行为取决于事务管理器)。

2. REQUIRES_NEW

  • 强制开启新事务,无论当前是否存在事务。
  • 必定触发事务创建,无论方法内容如何。

3. SUPPORTS

  • 如果当前存在事务,则加入;否则以非事务方式运行。
  • 若无数据库操作且无现有事务,不会创建事务

五、性能优化与验证

1. 如何验证事务是否开启?

  • 开启 DEBUG 日志
    logging.level.org.springframework.transaction=DEBUG
    logging.level.org.springframework.jdbc=DEBUG
    logging.level.org.springframework.orm.jpa=DEBUG
    
  • 观察日志中的事务生命周期事件(如 Creating new transactionCommitting transaction)。

2. 避免空事务的性能损耗

  • 移除不必要的 @Transactional 注解
  • 使用 @Transactional(propagation = Propagation.NOT_SUPPORTED) 显式禁用事务。

六、总结:事务开启的决策逻辑

条件事务是否开启?说明
方法标记为 @Transactional由代理对象和事务管理器触发,与数据库操作无关。
传播行为为 REQUIRES_NEW强制开启新事务。
传播行为为 SUPPORTS否(若无现有事务)仅在存在事务时加入。
使用 JDBC 事务管理器是(立即获取连接)即使无数据库操作,仍会获取连接并提交空事务。
使用 JPA 事务管理器是(延迟获取连接)事务被创建,但未实际获取连接(无数据库操作时)。

相关文章:

  • 【计算机网络】第1章:概述—分组延时、丢失和吞吐量
  • BugKu Web渗透之矛盾
  • 电脑长期不关机会怎样?
  • HarmonyOS 5 应用开发导读:从入门到实践
  • CloudCompare|点测量功能源码分析
  • ABP VNext + CRDT 打造实时协同编辑
  • 怎么快速判断一款MCU能否跑RTOS系统
  • OpenCV CUDA模块直方图计算------在 GPU 上计算图像直方图的函数calcHist()
  • XCTF-web-mfw
  • vue修改配置文件.env.development不生效
  • STM32:ESP8266 + MQTT 云端与报文全解析
  • MCP Server的五种主流架构:从原理到实践的深度解析
  • Vue Router 钩子函数与组件生命周期执行顺序详解
  • 【赵渝强老师】OceanBase的部署架构
  • GIT命令行的一些常规操作
  • HIS换代升级辅助脚手架:数据清洗、人员信息标准化、电子病历接口标准化、多模态影像接口标准化
  • 佰力博科技与您探讨DEAI介电阻抗分析仪的特点
  • 960g轻薄本,把科技塞进巧克力盒子
  • 数据分析学习笔记——A/B测试
  • PostgreSQL 数据完整性检查工具对比:amcheck 与 pg_checksums
  • 济南网站建设公司制作/免费发广告的网站
  • 网站难做/网站关键词怎么优化排名
  • dw 做网站模板/长尾关键词网站
  • 网站建设 内容缺乏/上海快速排名优化
  • 周口建设局网站/怎么从网上找国外客户
  • 织梦网站模板怎么做/只需要手机号的广告