基于Spring的aop实现多数据源动态切换
Spring多数据源管理类AbstractRoutingDataSource
实现AbstractRoutingDataSource重写determineCurrentLookupKey
首先配置项目的多数据源实体类交给spring管理
<beans>
<!-- 加载配置属性文件 -->
<context:property-placeholder ignore-unresolvable="true" location="classpath:application.properties" />
<!-- 数据源配置, 使用 BoneCP 数据库连接池 -->
<bean id="dataSourceMaster" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
<!-- 数据源驱动类可不写,Druid默认会自动根据URL识别DriverClass -->
<property name="driverClassName" value="${default.jdbc.driver}" />
<!-- 基本属性 url、user、password -->
<property name="url" value="${default.jdbc.url}" />
<property name="username" value="${default.jdbc.username}" />
<property name="password" value="${default.jdbc.password}" />
<!-- 配置初始化大小、最小、最大 -->
<property name="initialSize" value="${default.jdbc.pool.init}" />
<property name="minIdle" value="${default.jdbc.pool.minIdle}" />
<property name="maxActive" value="${default.jdbc.pool.maxActive}" />
<!-- 配置获取连接等待超时的时间 -->
<property name="maxWait" value="60000" />
<!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
<property name="timeBetweenEvictionRunsMillis" value="60000" />
<!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
<property name="minEvictableIdleTimeMillis" value="300000" />
<property name="validationQuery" value="${default.jdbc.testSql}" />
<property name="testWhileIdle" value="true" />
<property name="testOnBorrow" value="false" />
<property name="testOnReturn" value="false" />
<!-- 配置监控统计拦截的filters -->
<property name="filters" value="stat" />
</bean>
<!-- 分发库数据库 -->
<bean id="ddsDataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
<!-- 数据源驱动类可不写,Druid默认会自动根据URL识别DriverClass -->
<property name="driverClassName" value="${dds.jdbc.driver}" />
<!-- 基本属性 url、user、password -->
<property name="url" value="${dds.jdbc.url}" />
<property name="username" value="${dds.jdbc.username}" />
<property name="password" value="${dds.jdbc.password}" />
<!-- 配置初始化大小、最小、最大 -->
<property name="initialSize" value="${default.jdbc.pool.init}" />
<property name="minIdle" value="${default.jdbc.pool.minIdle}" />
<property name="maxActive" value="${default.jdbc.pool.maxActive}" />
<!-- 配置获取连接等待超时的时间 -->
<property name="maxWait" value="60000" />
<!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
<property name="timeBetweenEvictionRunsMillis" value="60000" />
<!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
<property name="minEvictableIdleTimeMillis" value="300000" />
<property name="validationQuery" value="${default.jdbc.testSql}" />
<property name="testWhileIdle" value="true" />
<property name="testOnBorrow" value="false" />
<property name="testOnReturn" value="false" />
</bean>
<!-- 可支持多数据源配置 -->
<bean id="dataSource" class="cn.angyi.jlkd.comm.mybatis.ThreadLocalRountingDataSource">
<property name="defaultTargetDataSource" ref="dataSourceMaster" />
<property name="targetDataSources">
<map key-type="cn.angyi.jlkd.comm.mybatis.DataSources">
<entry key="MASTER" value-ref="dataSourceMaster" />
<entry key="SLAVE" value-ref="ddsDataSource" />
</map>
</property>
</bean>
<aop:aspectj-autoproxy />
</beans>
在application.properties文件配置数据库参数
default.jdbc.type=
default.jdbc.driver=
default.jdbc.url=
default.jdbc.username=
default.jdbc.password=
default.jdbc.pool.init=1
default.jdbc.pool.minIdle=3
default.jdbc.pool.maxActive=20
default.jdbc.testSql=SELECT 'x' FROM DUAL
dds.jdbc.driver=
dds.jdbc.url=
dds.jdbc.username=
dds.jdbc.password=
数据源枚举类 DataSources
/**
* 数据源枚举类;
*/
public enum DataSources {
MASTER, SLAVE
}
自定义数据源管理类 DataSourceTypeManager
/**
* 数据源类型管理类;
*/
public class DataSourceTypeManager {
//ThreadLocal一般称为线程本地变量,它是一种特殊的线程绑定机制,将变量与线程绑定在一起,为每一个线程维护一个独立的变量副本 ThreadLocal可以理解为将对象的作用范围限制在一个线程上下文中,使得变量的作用域为“线程级”
private static final ThreadLocal<DataSources> dataSourceTypes = new ThreadLocal<DataSources>() {
@Override
protected DataSources initialValue() {
return DataSources.MASTER;
}
};
public static DataSources get() {
return dataSourceTypes.get();
}
public static void set(DataSources dataSourceType) {
dataSourceTypes.set(dataSourceType);
}
public static void reset() {
dataSourceTypes.set(DataSources.MASTER);
}
}
通过实现AbstractRoutingDataSource重写determineCurrentLookupKey切换数据源
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
/**
* 获取本地线程中保留的数据源;
*/
public class ThreadLocalRountingDataSource extends AbstractRoutingDataSource {
@Override
protected Object determineCurrentLookupKey() {
return DataSourceTypeManager.get();
}
}
最后,通过aop切面动态切换数据源,这里我用的注解的方式,其他的方法请自行百度
注解类 DataSource
import java.lang.annotation.*;
@Documented
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface DataSource {
// 这个配置了默认数据源SLAVE
DataSources targetDataSurce() default DataSources.SLAVE;
}
根据aop动态修改数据源
@Aspect
@Component
@Order(1)
public class DataSourcesChange {
private static final Logger logger = LoggerFactory.getLogger(DataSource.class);
@Pointcut("@annotation(这里是上面注解的路径名)")
public void point() {
}
@Before(value = "point()")
public void before(JoinPoint jp) {
Class<? extends Object> targetClass = jp.getTarget().getClass();
String methodName = jp.getSignature().getName(); //方法名称
Class[] parameterTypes = ((MethodSignature)jp.getSignature()).getParameterTypes(); //方法参数类型列表
logger.debug("========== "+targetClass.getName()+"."+methodName+" ==========");
Method method=null;
try {
method = targetClass.getMethod(methodName, parameterTypes); //获取方法
} catch (Exception e) {
logger.error("========== sourceBefore get method error ==========" ,e);
}
if(method!=null && method.isAnnotationPresent(DataSource.class)){
DataSource da = method.getAnnotation(DataSource.class);
DataSources dataSurce = da.targetDataSurce();
DataSourceTypeManager.set(dataSurce);
logger.debug("========== set dataSource "+dataSurce+" ==========");
}
}
@After(value = "point()")
public void after(JoinPoint jp) {
DataSourceTypeManager.set(DataSources.MASTER);
}
}
之后只要在方法中添加注解
@DataSource(targetDataSurce = DataSources.SLAVE)就可以了
也可以使用@DataSource 等价于 @DataSource(targetDataSurce = DataSources.MASTER)