Spring多数据源环境下的事务与数据源切换

核心问题说明

注意事项:

同一个事务下是无法切换数据源

禁止父方法使用@Transactional创建事务,子方法使用@DataSource切换数据源

正确用法: 子方法单独创建事务或父方法使用@Transactional(propagation = Propagation.REQUIRES_NEW)为所有子方法创建新事务

这个注意事项涉及Spring中事务管理和动态数据源切换的关键约束,让我深入解释这个问题。

为什么同一事务下无法切换数据源?

技术原理

  1. 事务与连接的绑定关系
  • 当事务开始时,Spring会从当前数据源获取一个数据库连接
  • 该连接会被绑定到当前线程的事务资源中(ThreadLocal)
  • 事务管理器使用这个固定的连接执行所有SQL操作
  • 数据源切换的时机
  • 数据源切换需要在获取连接前完成
  • 一旦事务开启并获取连接,数据源已确定且不可更改
  • 切换数据源实际是更换连接工厂,而非已获取的连接

问题场景示例

错误示例1:事务中尝试切换数据源

复制代码
@Service

public class UserService {

    @Autowired

    private OrderService orderService;

    @Transactional  // 主数据源的事务

    public void processUserOrder(Long userId) {

        // 使用主数据源操作用户数据

        updateUserStatus(userId);

        // 尝试在同一事务中切换数据源 - 这将失败!

        orderService.createOrder(userId);

    }

}

@Service

public class OrderService {

    @DataSource(DataSourceType.SLAVE)  // 尝试切换到从数据源

    public void createOrder(Long userId) {

        // 此方法实际上仍会使用主数据源,因为它在父事务中执行

        saveOrder(new Order(userId));

    }

}

问题解释:

  • 当processUserOrder方法开始时,Spring创建一个事务并从主数据源获取连接
  • 调用createOrder时,虽然标注了@DataSource,但由于已在事务中,数据源切换不会生效
  • 所有操作都会使用主数据源的连接

错误示例2:跨数据源的分布式事务期望

复制代码
@Transactional

@DataSource(DataSourceType.MASTER)

public void transferMoney() {

    // 主库操作

    accountDao.deductFromMaster(100);

    // 错误:期望切换到从库并在同一事务中操作

    @DataSource(DataSourceType.SLAVE)

    accountDao.depositToSlave(100);

    // 如果出错希望两边都回滚,但实际上这是不可能的

}

正确的解决方案

方案1:为子方法创建独立事务

复制代码
@Service

public class UserService {

    @Autowired

    private OrderService orderService;

    // 不在父方法开启事务

    public void processUserOrder(Long userId) {

        // 主数据源操作,单独事务

        updateUserWithTransaction(userId);

        // 从数据源操作,另一个独立事务

        orderService.createOrder(userId);

    }

    @Transactional

    public void updateUserWithTransaction(Long userId) {

        updateUserStatus(userId);

    }

}

@Service

public class OrderService {

    @DataSource(DataSourceType.SLAVE)

    @Transactional  // 独立事务

    public void createOrder(Long userId) {

        // 现在使用的是从数据源,且在独立事务中

        saveOrder(new Order(userId));

    }

}

方案2:使用REQUIRES_NEW传播行为

复制代码
@Service

public class UserService {

    @Autowired

    private OrderService orderService;

    @Transactional  // 主数据源事务

    public void processUserOrder(Long userId) {

        // 使用主数据源

        updateUserStatus(userId);

        // 子方法会挂起当前事务并创建新事务

        orderService.createOrder(userId);

    }

}

@Service

public class OrderService {

    @DataSource(DataSourceType.SLAVE)

    @Transactional(propagation \= Propagation.REQUIRES\_NEW)  // 关键点:创建新事务

    public void createOrder(Long userId) {

        // 此方法会挂起父事务,创建新事务,并使用从数据源

        saveOrder(new Order(userId));

    }

}
相关推荐
麦兜*4 分钟前
Docker 部署 MongoDB:单节点与副本集的最佳实践
java·spring boot·mongodb·spring cloud·docker·容器·maven
小小怪KO7 分钟前
分布式锁解决集群下一人一单超卖问题
java·分布式·tomcat·后端开发·实习·黑马点评
格林威20 分钟前
Linux使用-MySQL的使用
linux·运维·人工智能·数码相机·mysql·计算机视觉·视觉检测
智码看视界22 分钟前
老梁聊全栈系列:(阶段一)从单体到云原生的演进脉络
java·云原生·c5全栈
望获linux41 分钟前
【实时Linux实战系列】规避缺页中断:mlock/hugetlb 与页面预热
java·linux·服务器·数据库·chrome·算法
小程序设计1 小时前
【springboot+vue】高校迎新平台管理系统(源码+文档+调试+基础修改+答疑)
vue.js·spring boot·后端
longerxin20201 小时前
MongoDB 在线安装-一键安装脚本(CentOS 7.9)
数据库·mongodb·centos
失散131 小时前
分布式专题——9 Redis7底层数据结构解析
java·数据结构·redis·分布式·缓存·架构
馨谙1 小时前
设计模式之单例模式大全---java实现
java·单例模式·设计模式
程序员TNT1 小时前
Shoptnt 安全架构揭秘:JWT 认证与分布式实时踢人方案
java·redis·分布式·架构