mybatisplus手动切换数据源

// 切换数据源 slave 为配置的名称
DynamicDataSourceContextHolder.push("slave");
// TODO 业务代码
...
// 清理
DynamicDataSourceContextHolder.clear();

详情看
com.baomidou.dynamic.datasource.ds.AbstractRoutingDataSource

### MyBatisPlus 结合 Druid 实现多数据源的配置和使用 在 Java 开发中,MyBatis-Plus 是一款增强型的 MyBatis 框架,而 Druid 则是一款高效的数据库连接池工具。两者结合可以实现更灵活的数据访问功能[^2]。 #### 配置多数据源的核心思路 为了支持多个数据源,通常需要通过动态切换的方式完成操作。以下是具体的实现方式: 1. **引入依赖** 在项目中添加必要的 Maven 或 Gradle 依赖项来集成 MyBatis-Plus 和 Druid。 ```xml <!-- MyBatis Plus --> <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.5.1</version> </dependency> <!-- Druid 数据库连接池 --> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid-spring-boot-starter</artifactId> <version>1.2.8</version> </dependency> ``` 2. **定义数据源配置类** 创建一个配置类用于管理不同的数据源实例,并将其注册到 Spring 容器中。 ```java import com.alibaba.druid.pool.DruidDataSource; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Primary; @Configuration public class DataSourceConfig { @Bean(name = "dataSourceOne") @ConfigurationProperties(prefix = "spring.datasource.one") // 对应 application.yml 中的配置键名 public DruidDataSource dataSourceOne() { return new DruidDataSource(); } @Bean(name = "dataSourceTwo") @ConfigurationProperties(prefix = "spring.datasource.two") public DruidDataSource dataSourceTwo() { return new DruidDataSource(); } } ``` 3. **创建动态数据源路由逻辑** 动态数据源的关键在于自定义一个 `AbstractRoutingDataSource` 的子类,该类负责根据上下文决定当前使用的具体数据源。 ```java import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; public class DynamicDataSource extends AbstractRoutingDataSource { private static final ThreadLocal<String> contextHolder = new ThreadLocal<>(); public static void setDataSource(String key) { contextHolder.set(key); } public static String getDataSource() { return contextHolder.get(); } public static void clearDataSource() { contextHolder.remove(); } @Override protected Object determineCurrentLookupKey() { return getDataSource(); } } ``` 4. **绑定数据源并启用事务管理** 将之前定义好的多个数据源注入到动态数据源对象中,并设置为主数据源。 ```java import javax.sql.DataSource; import java.util.HashMap; import java.util.Map; @Configuration public class DynamicDataSourceConfig { @Autowired @Qualifier("dataSourceOne") private DataSource dataSourceOne; @Autowired @Qualifier("dataSourceTwo") private DataSource dataSourceTwo; @Primary @Bean(name = "dynamicDataSource") public DynamicDataSource dynamicDataSource() { DynamicDataSource dynamicDataSource = new DynamicDataSource(); Map<Object, Object> targetDataSources = new HashMap<>(); targetDataSources.put("one", dataSourceOne); targetDataSources.put("two", dataSourceTwo); dynamicDataSource.setTargetDataSources(targetDataSources); dynamicDataSource.setDefaultTargetDataSource(dataSourceOne); // 默认使用第一个数据源 return dynamicDataSource; } } ``` 5. **编写拦截器控制数据源切换** 使用 AOP 技术或者手动调用 API 来指定每次查询应该走哪个数据源。 ```java @Aspect @Component public class DataSourceInterceptor { @Before("@annotation(com.example.annotation.TargetDataSource)") public void changeDataSource(JoinPoint point) throws Exception { MethodSignature signature = (MethodSignature) point.getSignature(); TargetDataSource annotation = signature.getMethod().getAnnotation(TargetDataSource.class); if (annotation != null && StringUtils.isNotEmpty(annotation.value())) { DynamicDataSource.setDataSource(annotation.value()); } } @After("@annotation(com.example.annotation.TargetDataSource)") public void restoreDataSource(JoinPoint point) { DynamicDataSource.clearDataSource(); // 清理线程变量中的状态 } } ``` 6. **测试验证** 编写单元测试案例以确认不同场景下能否正常切换至对应的目标表单。 --- 关于 `NULL` 的问题,在 C/C++ 中确实存在一些特殊情况处理机制。虽然表面上看 `(void*)0` 可能会被认为等价于标准意义上的空指针常量表达形式之一,但实际上这取决于编译环境以及内部实现细节差异所致[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值