在Spring Boot + MyBatis-Plus 中实现动态数据源切换,可以使用AbstractRoutingDataSource和AOP。
以下是核心步骤和示例代码:
- 创建动态数据源(DynamicDataSource)类,继承
AbstractRoutingDataSource
,并重写determineCurrentLookupKey
方法。
public class DynamicDataSource extends AbstractRoutingDataSource {
@Override
protected Object determineCurrentLookupKey() {
return DataSourceContextHolder.getDataSourceType();
}
}
- 使用ThreadLocal保存当前数据源标识。
public class DataSourceContextHolder {
private static final ThreadLocal<String> contextHolder = new ThreadLocal<>();
public static void setDataSourceType(String dataSourceType) {
contextHolder.set(dataSourceType);
}
public static String getDataSourceType() {
return contextHolder.get();
}
public static void clearDataSourceType() {
contextHolder.remove();
}
}
- 创建AOP切面,在切面方法执行前切换数据源。
@Aspect
@Component
public class DataSourceAspect {
@Before("@annotation(targetDataSource)")
public void switchDataSource(JoinPoint point, TargetDataSource targetDataSource) {
DataSourceContextHolder.setDataSourceType(targetDataSource.value());
}
@After("@annotation(targetDataSource)")
public void restoreDataSource(JoinPoint point, TargetDataSource targetDataSource) {
DataSourceContextHolder.clearDataSourceType();
}
}
- 创建自定义注解
TargetDataSource
用于标注方法的数据源。
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface TargetDataSource {
String value();
}
- 配置动态数据源,并将其注册到Spring上下文中。
@Configuration
public class DataSourceConfig {
@Bean
public DataSource dataSource() {
DynamicDataSource dynamicDataSource = new DynamicDataSource();
dynamicDataSource.setDefaultTargetDataSource(primaryDataSource()); // 默认数据源
// 配置多数据源
Map<Object, Object> dataSourceMap = new HashMap<>();
dataSourceMap.put("primary", primaryDataSource());
dataSourceMap.put("secondary", secondar