1.首先pom文件
mysql mysql-connector-javacom.baomidou mybatis-plus-boot-startercom.baomidou dynamic-datasource-spring-boot-starter
2.数据源公用类
import com.alibaba.druid.pool.DruidDataSource;
import lombok.Data;
import org.springframework.stereotype.Component;
@Component
@Data
public class DatasourceBuilder {
private String driverClassName;
private String url;
private String username;
private String password;
public DruidDataSource createDruidDataSource() {
DruidDataSource datasource = new DruidDataSource();
datasource.setDriverClassName(driverClassName);
datasource.setUrl(url);
datasource.setUsername(username);
datasource.setPassword(password);
return datasource;
}
}
3.
@Configuration
@ConfigurationProperties(prefix = "spring.read.one")
public class ReadoneDataSourceBuilder extends DatasourceBuilder {
}
@Configuration
@ConfigurationProperties(prefix = "spring.write.datasource")
public class WriteDataSourceBuilder extends DatasourceBuilder {
}
4.配置多数据源
@Configuration
public class DruidConfiguration {
@Autowired
private MyMybatisProperties mybatisProperties;
@Autowired
private WriteDataSourceBuilder writeDataSourceBuilder;
@Autowired
private ReadoneDataSourceBuilder readOneDataSourceBuilder;
@Autowired
private DataSourceConfigBuilder dataSourceConfigBuilder;
@Bean(name = "writeDataSource")
@Primary
public DataSource writeDataSource() {
DruidDataSource dataSource = writeDataSourceBuilder.createDruidDataSource();
return dataSourceConfigBuilder.config(dataSource);
}
@Bean(name = "readDataSourceOne")
public DataSource readDataSourceOne() {
DruidDataSource dataSource = readOneDataSourceBuilder.createDruidDataSource();
return dataSourceConfigBuilder.config(dataSource);
}
@Bean("readDataSources")
public List readDataSources() {
List dataSources = new ArrayList();
dataSources.add(readDataSourceOne());
return dataSources;
}
@Bean
public RoutingDataSource dynamicDataSource() {
return new RoutingDataSource(writeDataSource(),readDataSources());
}
@Bean
@ConditionalOnMissingBean
public SqlSessionFactory sqlSessionFactory() throws Exception {
Resource[] resources = new PathMatchingResourcePatternResolver().getResources(mybatisProperties.getMapperLocations());
// 解决myBatis下 不能嵌套jar文件的问题
VFS.addImplClass(SpringBootVFS.class);
MybatisSqlSessionFactoryBean factory = new MybatisSqlSessionFactoryBean();
factory.setDataSource(dynamicDataSource());
factory.setTypeAliasesPackage(mybatisProperties.getTypeAliasesPackage());
factory.setMapperLocations(resources);
SqlSessionFactory sqlSessionFactory = factory.getObject();
if (sqlSessionFactory != null) {
sqlSessionFactory.getConfiguration().setMapUnderscoreToCamelCase(true);
}
return sqlSessionFactory;
}
@Bean
public DataSourceTransactionManager transactionManager(AbstractRoutingDataSource source) {
return new DataSourceTransactionManager(source);
}
5.AOP加本地线程变量
@Aspect
@Component
@Slf4j
@Order(0)
public class DatasourceAop {
private final String[] readSuffix = {"read", "get", "find", "search"};
@Before("execution(* com.tvunetworks.invoice.service..*.*(..))")
public void switchDataSourceType(final JoinPoint joinPoint) {
//判断是否是Transactional方法
if (joinPoint.getSignature() instanceof MethodSignature) {
Method method = ((MethodSignature) joinPoint
.getSignature()).getMethod();
if (method.isAnnotationPresent(Transactional.class)) {
DynamicDataSourceContextHolder.push(DataSourceType.write.getType());
return;
}
}
for (String read : readSuffix) {
String name = joinPoint.getSignature().getName().toLowerCase();
if (name.startsWith(read)) {
log.debug("dataSource switch to: Read");
DynamicDataSourceContextHolder.push(DataSourceType.read.getType());
return;
}
}
log.debug("dataSource switch to: Write");
DynamicDataSourceContextHolder.push(DataSourceType.write.getType());
}
}
public class DataSourceContextHolder {
private static final ThreadLocal LOCAL = new ThreadLocal();
public static ThreadLocal getLocal() {
return LOCAL;
}
public static void read() {
LOCAL.set(DataSourceType.read.getType());
}
public static void write() {
LOCAL.set(DataSourceType.write.getType());
}
public static String getJdbcType() {
String s = LOCAL.get();
LOCAL.remove();
return s;
}
}
切换数据源
public class RoutingDataSource extends AbstractRoutingDataSource {
private final int readSize;
private final AtomicInteger count=new AtomicInteger(0);
public RoutingDataSource(DataSource defaultTargetDataSource, List targetDataSources){
Map 


