三分钟知识点:Spring事务的传播机制

三分钟知识点:Spring事务的传播机制

三分钟,复习一个知识点!

一、Spring事务管理

Spring通过声明式事务管理简化了数据库事务操作,核心控制要素包含传播机制(Propagation)和隔离级别(Isolation)。

二、核心机制说明

Spring的事务管理基于代理模式 实现,当方法被@Transactional注解时:

  1. 事务边界由代理对象控制
  2. 事务上下文通过线程绑定传播
  3. 同类内部方法调用会绕过代理

三、事务传播机制

1. REQUIRED(默认)

行为:当前存在事务则加入,否则新建事务

java 复制代码
@Service
public class OrderService {
    @Transactional(propagation = Propagation.REQUIRED)
    public void createOrder() {
        // 主业务逻辑
        userService.updateUserPoints();
    }
}

@Service
public class UserService {
    @Transactional(propagation = Propagation.REQUIRED)
    public void updateUserPoints() {
        // 更新用户积分
    }
}

效果:当createOrder()调用updateUserPoints()时,两个方法共享同一个事务,任一操作失败将整体回滚


2. SUPPORTS

行为:存在事务则加入,否则以非事务方式运行

java 复制代码
@Transactional(propagation = Propagation.SUPPORTS)
public void getOrderDetail(Long orderId) {
    // 查询订单详情
}

效果:当被事务方法调用时加入事务,被非事务方法调用时无事务保护


3. MANDATORY

行为:强制要求存在事务,否则抛出异常

java 复制代码
@Transactional(propagation = Propagation.MANDATORY)
public void auditOrder(Long orderId) {
    // 订单审核逻辑
}

效果:非事务上下文调用此方法将抛出IllegalTransactionStateException


4. REQUIRES_NEW

行为:始终新建独立事务,挂起当前事务

java 复制代码
@Service
public class LogService {
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void saveOperationLog() {
        // 记录操作日志
    }
}

效果:日志记录与主事务完全独立,主事务回滚不影响日志事务


5. NOT_SUPPORTED

行为:以非事务方式执行,挂起当前事务

java 复制代码
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void generateReport() {
    // 生成复杂报表
}

效果:强制执行非事务操作,适用于耗时只读操作


6. NEVER

行为:强制要求无事务环境,存在事务则抛出异常

java 复制代码
@Transactional(propagation = Propagation.NEVER)
public void cacheRefresh() {
    // 刷新本地缓存
}

效果:防止缓存操作与事务产生关联


7. NESTED

行为:在现有事务中创建嵌套事务(使用保存点)

java 复制代码
@Transactional(propagation = Propagation.NESTED)
public void updateInventory() {
    // 库存扣减逻辑
}

效果:嵌套事务回滚不影响外层事务,外层事务回滚将导致嵌套事务回滚


四、事务隔离级别

1. READ_UNCOMMITTED

java 复制代码
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
public void processData() {
    // 可能读取到未提交数据
}

问题:允许脏读、不可重复读、幻读


2. READ_COMMITTED

java 复制代码
@Transactional(isolation = Isolation.READ_COMMITTED)
public void updateRecord() {
    // 只能看到已提交数据
}

特点:防止脏读,允许不可重复读和幻读


3. REPEATABLE_READ

java 复制代码
@Transactional(isolation = Isolation.REPEATABLE_READ)
public void batchProcess() {
    // 保证重复读取一致性
}

优势:防止脏读和不可重复读


4. SERIALIZABLE

java 复制代码
@Transactional(isolation = Isolation.SERIALIZABLE)
public void financialSettlement() {
    // 最高级别隔离
}

代价:完全串行化,性能影响最大


五、配置建议

传播机制选择原则

  • 数据一致性优先:REQUIRED
  • 独立业务操作:REQUIRES_NEW
  • 非关键操作:NOT_SUPPORTED
  • 审计日志场景:NESTED

隔离级别权衡

  • 常规场景:READ_COMMITTED
  • 财务系统:REPEATABLE_READ
  • 高并发写入:慎用SERIALIZABLE
properties 复制代码
# 全局默认配置
spring.transaction.default-timeout=30
spring.jpa.properties.hibernate.connection.isolation=2 # READ_COMMITTED

六、常见问题排查

  1. 事务不生效检查点

    • 是否启用@EnableTransactionManagement
    • 异常类型是否触发回滚(默认仅RuntimeException)
    • 是否同类方法调用(代理失效问题)
  2. 性能优化建议

    • 合理缩短事务边界
    • 只读事务标记readOnly=true
    • 避免长事务持有数据库连接

通过合理配置传播机制和隔离级别,可以在保证数据一致性的同时实现最佳性能平衡。建议通过集成测试验证事务行为,特别是在复杂业务场景中。


七、最佳实践建议

  1. 事务边界设计原则
    • 保持事务方法短小
    • 避免在事务方法中处理复杂业务逻辑
    • 事务方法应聚焦数据库操作

最后

如果文章对你有帮助,点个免费的赞鼓励一下吧!关注gzh:加瓦点灯, 每天推送干货知识!

相关推荐
rannn_11123 分钟前
【苍穹外卖|Day4】套餐页面开发(新增套餐、分页查询、删除套餐、修改套餐、起售停售)
java·spring boot·后端·学习
短剑重铸之日33 分钟前
《设计模式》第十一篇:总结
java·后端·设计模式·总结
Dragon Wu2 小时前
Spring Security Oauth2.1 授权码模式实现前后端分离的方案
java·spring boot·后端·spring cloud·springboot·springcloud
一个有梦有戏的人2 小时前
Python3基础:进阶基础,筑牢编程底层能力
后端·python
爬山算法2 小时前
Hibernate(88)如何在负载测试中使用Hibernate?
java·后端·hibernate
独断万古他化2 小时前
【Spring 原理】Bean 的作用域与生命周期
java·后端·spring
我爱加班、、3 小时前
Websocket能携带token过去后端吗
前端·后端·websocket
一 乐3 小时前
校园二手交易|基于springboot + vue校园二手交易系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端
80530单词突击赢3 小时前
SpringBoot整合SpringMVC全解析
java·spring boot·后端
hdsoft_huge3 小时前
1panel面板中部署SpringBoot和Vue前后端分离系统 【图文教程】
vue.js·spring boot·后端