聊聊@DSTransactional的坑

为什么写这篇文章

最近项目里面使用了多数据源功能,使用的包是

xml 复制代码
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
    <version>3.5.1</version>
</dependency>

使用过程中就出现一个问题,在一个@DSTransactional注解里面标注的方法,前面插入的数据后面却查不出来。经过一番测试,终于找到问题的根源,特此记录以备后续查阅

问题复现

配置如下

yml 复制代码
spring:
  datasource:
    dynamic:
      primary: yrt
      strict: false
      datasource:
        yrt:
          url: xxx
          username: xxx
          password: xxx
        ent:
          url: xxx
          username: xxx
          password: xxx

为了打印数据源切换的日志,还增加了如下配置

yml 复制代码
logging:
  config: classpath:logback.xml
  level:
    org.springframework.transaction: DEBUG
    org.springframework.jdbc.datasource: DEBUG
    com.baomidou.dynamic.datasource: DEBUG

坑描述

写了@DS和没写@DS的类中的方法,@DS("primary")@DS("yrt"),即使primary对应的数据库就是yrt也会导致事务切换,事务切换就意味着前面插入的代码在后面无法查询,下面给出列子

代码描述

DsTransactionalTest

java 复制代码
public class DsTransactionalTest {

    @Autowired
    DsTransactionalTestService dsTransactionalTestService;

    @org.junit.Test
    public void testDsTransactional() {
        dsTransactionalTestService.dsTransactionalTest();
    }
}

DsTransactionalTestService

java 复制代码
@Service
@AllArgsConstructor
public class DsTransactionalTestService {

    private final VendorMPService vendorMPService;
    private final DealerMPService dealerMPService;

    @DSTransactional
    public void dsTransactionalTest() {
        Integer vid = 40048;
        vendorMPService.saveTest(vid);
        dealerMPService.testQueryVendor(vid);
    }
}

VendorMPService

java 复制代码
@Service
public class VendorMPService extends ServiceImpl<VendorMPMapper, VendorMP> {
    public void saveTest(Integer vid) {
        VendorMP vendorMP = new VendorMP();
        vendorMP.setVendorInnerSn(vid);
        vendorMP.setName("测试厂商" + vid);
        // 省略其他字段
        this.save(vendorMP);
    }
}

DealerMPService

java 复制代码
@Service
@AllArgsConstructor
@DS("primary")
public class DealerMPService extends ServiceImpl<DealerMPMapper, DealerMP> {
    private final VendorMPService vendorMPService;

    public void testQueryVendor(Integer vid) {
        System.out.println("厂商信息:" + vendorMPService.getById(vid));
    }
}

此时安装我们的理解,应该后面要能查询出来数据才对,但运行结果如下

虽然有inser语句,但是并没有查询出来数据,那么此时我们来修改代码,把@DS("primary") 也在VendorMPService 上面新增一份,就能查出来数据了

java 复制代码
@Service
@DS("primary")
public class VendorMPService extends ServiceImpl<VendorMPMapper, VendorMP> {
// 省略
}


我比较懒 篇幅所限,另一种情况,一个标注的是@DS("primary"),另一个标注的是@DS("yrt"),即使primary对应的数据源就是yrt 也是跟上面的情况是一样的,这里就不贴代码了。

回滚情况

我测试过程中,如果在后面的方法报错,前面的插入会回滚,即使一个标注的是@DS("primary"),另一个标注的是@DS("yrt"),或者一个标注了@DS("primary")另一个没标,都会回滚,也就是说:查询的时候虽然查询不出来,但是报错时能正常回滚。这就很奇怪,猜测是虽然有事务的切换,但是事务的ID是一样的,所以本质上还是同一个事务。

总结或最佳实践

  1. 理论上不要出现前面插入后面又查询是最好,直接把插入的对象传值到后面就行了,但是在实际项目大了以后,方法之间的调用错综复杂,有时候会经常出现开发某个新方法,但是有部分逻辑过于复杂进而直接调用的情况,如果项目对@DSTransactional@DS注解的使用不够熟练,没有做好规定,很容易会引发这个BUG。
  2. 对于@DS的使用,要规范团队统一使用如@DS("primary"), 并且对于领域范围确定的service, 都要标注这个 @DS("primary"),不要一部分标注一部分不标,这样应该可以减少BUG
  3. 千万不要在 @DSTransactional 里面调用 @Transactional 标注的方法,两种注解混着用,得到的结果很奇怪,我目前还没测出会有什么奇怪的情况~,但是不要混着用时对的。
  4. 非必要时不要使用 @DSTransactional ,如果在某个方法里面需要新增A数据库的数据,然后查询B数据库的数据,此时直接给方法标注为@Transactional 然后在查询B数据库的时候,使用@Transactional(propagation = Propagation.NOT_SUPPORTED, rollbackFor = Exception.class),即事务不传播。
相关推荐
鸡蛋灌Bean8 小时前
mybatis分页深入了解
java·数据库·mybatis
野生技术架构师8 小时前
Tomcat Service的设计和实现:StandardService
java·tomcat
Gofarlic_OMS8 小时前
UG/NX许可证管理高频技术问题解答汇编
java·大数据·运维·服务器·汇编·人工智能
逐星ing8 小时前
IDEA 无法识别 `mvn install` 最新 SNAPSHOT 依赖的根因与完整解决方案
java·ide·intellij-idea
流觞 无依8 小时前
Spring Boot 未授权访问漏洞排查与修复指南
java·spring boot·后端
Java开发的小李8 小时前
SpringBoot 高流量高并发 基础全面讲解
java·spring boot·后端·性能优化
随风,奔跑8 小时前
Spring Cloud Alibaba(六)-链路追踪SkyWalking
java·后端·spring·skywalking
wuminyu8 小时前
专家视角看Lambda表达式的原理解析
java·linux·c语言·jvm·c++
wangbing11258 小时前
Java处理csv文件总是丢数据
java·开发语言·python
云烟成雨TD8 小时前
Spring AI 1.x 系列【30】向量数据库:核心 API 和入门案例
java·人工智能·spring