最近因为项目需要在做两个项目间数据同步的需求,具体是项目1的数据通过消息队列同步到项目2中,因为这个更新操作还涉及到更新多个库的数据,所以就需要多数据源切换的操作。下面就讲讲在Spring中如何进行数据源切换。这里是使用AbstractRoutingDataSource类来完成具体的操作,AbstractRoutingDataSource是Spring2.0后增加的。
实现数据源切换的功能就是自定义一个类扩展AbstractRoutingDataSource抽象类,其实该相当于数据源DataSourcer的路由中介,可以实现在项目运行时根据相应key值切换到对应的数据源DataSource上。先看看AbstractRoutingDataSource的源码:
public abstract class AbstractRoutingDataSource extends AbstractDataSource implements InitializingBean {
/* 只列出部分代码 */
private Map<Object, Object> targetDataSources;
private Object defaultTargetDataSource;
private boolean lenientFallback = true;
private DataSourceLookup dataSourceLookup = new JndiDataSourceLookup();
private Map<Object, DataSource> resolvedDataSources;
private DataSource resolvedDefaultDataSource;
@Override
public Connection getConnection() throws SQLException {
return determineTargetDataSource().getConnection();
}
@Override
public Connection getConnection(String username, String password) throws SQLException {
return determineTargetDataSource().getConnection(username, password);
}
protected DataSource determineTargetDataSource() {
Assert.notNull(this.resolvedDataSources, "DataSource router not initialized");
Object lookupKey = determineCurrentLookupKey();
DataSource dataSource = this.resolvedDataSources.get(lookupKey);
if (dataSource == null && (this.lenientFallback || lookupKey == null)) {
dataSource = this.resolvedDefaultDataSource;
}
if (dataSource == null) {
throw new IllegalStateException("Cannot determine target DataSource for lookup key [" + lookupKey + "]");
}
return dataSource;
}
protected abstract Object determineCurrentLookupKey();
}
从源码可以看出AbstractRoutingDataSource继承了AbstractDataSource并实现了InitializingBean,AbstractRoutingDataSource的getConnection()方法调用了determineTargetDataSource()的该方法,这里重点看determineTargetDataSource()方法代码,方法里使用到了determineCurrentLookupKey()方法,它是AbstractRoutingDataSource类的抽象方法,也是实现数据源切换要扩展的方法,该方法的返回值就是项目中所要用的DataSource的key值,拿到该key后就可以在resolvedDataSource中取出对应的DataSource,如果key找不到对应的DataSource就使用默认的数据源。
自定义类扩展AbstractRoutingDataSource类时就是要重写determineCurrentLookupKey()方法来实现数据源切换功能。下面是自定义的扩展AbstractRoutingDataSource类的实现:
/**
* 获得数据源
*/
public class MultipleDataSource extends AbstractRoutingDataSource{
@Override
protected Object determineCurrentLookupKey() {
return DynamicDataSourceHolder.getRouteKey();
}
}
DynamicDataSourceHolder类如下,实现对数据源的操作功能:
/**
* 数据源操作类
*/
public class DynamicDataSourceHolder {
private static ThreadLocal<String> routeKey = new ThreadLocal<String>();
/**
* 获取当前线程的数据源路由的key
*/
public static String getRouteKey()
{
String key = routeKey.get();
return key;
}
/**
* 绑定当前线程数据源路由的key
* 使用完成后必须调用removeRouteKey()方法删除
*/
public static void setRouteKey(String key)
{
routeKey.set(key);
}
/**
* 删除与当前线程绑定的数据源路由的key
*/
public static void removeRouteKey()
{
routeKey.remove();
}
}
下面在xml文件中配置多个数据源:
<!-- 数据源 -->
<bean id="dataSource1" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="net.sourceforge.jtds.jdbc.Driver">
</property>
<property name="url" value="jdbc:jtds:sqlserver://127.0.0.1;databaseName=test">
</property>
<property name="username" value="***"></property>
<property name="password" value="***"></property>
</bean>
<bean id="dataSource2" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="net.sourceforge.jtds.jdbc.Driver">
</property>
<property name="url" value="jdbc:jtds:sqlserver://127.0.0.2:1433;databaseName=test">
</property>
<property name="username" value="***"></property>
<property name="password" value="***"></property>
</bean>
<!-- 配置多数据源映射 -->
<bean id="multipleDataSource" class="MultipleDataSource" >
<property name="targetDataSources">
<map key-type="java.lang.String">
<entry value-ref="dataSource1" key="dataSource1"></entry>
<entry value-ref="dataSource2" key="dataSource2"></entry>
</map>
</property>
<!-- 默认数据源 -->
<property name="defaultTargetDataSource" ref="dataSource1" >
</property>
</bean>
到这里基本的配置就完成了,下面只要在需要切换数据源的地方调用方法就行了,一般是在dao层操作数据库前进行切换的,只需在数据库操作前加上如下代码即可:
DynamicDataSourceHolder.setRouteKey("dataSource2");
上面介绍的是在dao层当需要切换数据源时手动加上切换数据源的代码,也可以使用AOP的方式,把配置的数据源类型都设置成注解标签,在dao层中需要切换数据源操作的方法或类上写上注解标签,这样实现起来可操作性也更强。
@DataSourceKey("dataSource1")
public interface TestEntityMapper extends MSSQLMapper<TestEntity> {
public void insertTest(TestEntity testEntity);
}
DataSourceKey注解代码如下:
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface DataSourceKey {
String value() default "";
}
注解配置完后就要写一个实现数据源切换的类,如下:
public class MultipleDataSourceExchange {
/**
* 拦截目标方法,获取由@DataSource指定的数据源标识,设置到线程存储中以便切换数据源
*/
public void beforeDaoMethod(JoinPoint point) throws Exception {
Class<?> target = point.getTarget().getClass();
MethodSignature signature = (MethodSignature) point.getSignature();
// 默认使用目标类型的注解,如果没有则使用其实现接口的注解类
for (Class<?> cls : target.getInterfaces()) {
resetDataSource(cls, signature.getMethod());
}
resetDataSource(target, signature.getMethod());
}
/**
* 提取目标对象方法注解和类注解中的数据源标识
*/
private void resetDataSource(Class<?> cls, Method method) {
try {
Class<?>[] types = method.getParameterTypes();
// 默认使用类注解
if (cls.isAnnotationPresent(DataSourceKey.class)) {
DataSourceKey source = cls.getAnnotation(DataSourceKey.class);
DynamicDataSourceHolder.setRouteKey(source.value());
}
// 方法注解可以覆盖类注解
Method m = cls.getMethod(method.getName(), types);
if (m != null && m.isAnnotationPresent(DataSourceKey.class)) {
DataSourceKey source = m.getAnnotation(DataSourceKey.class);
DynamicDataSourceHolder.setRouteKey(source.value());
}
} catch (Exception e) {
System.out.println(cls + ":" + e.getMessage());
}
}
}
代码写完后就要在xml配置文件上添加配置了(只列出部分配置):
<bean id="multipleDataSourceExchange" class="MultipleDataSourceExchange "/>
<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="multipleDataSource" />
</bean>
<tx:advice id="txAdvice" transaction-manager="txManager">
<tx:attributes>
<tx:method name="insert*" propagation="NESTED" rollback-for="Exception"/>
<tx:method name="add*" propagation="NESTED" rollback-for="Exception"/>
...
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="service" expression="execution(* com.datasource..*.service.*.*(..))"/>
<!-- 注意切换数据源操作要比持久层代码先执行 -->
<aop:advisor advice-ref="multipleDataSourceExchange" pointcut-ref="service" order="1"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="service" order="2"/>
</aop:config>
到此就完成使用AOP的方式实现多数据源的动态切换了。
# spring
# 多数据源切换
# spring切换数据源
# spring动态切换数据源
# Spring AbstractRoutingDatasource 动态数据源的实例讲解
# 浅谈利用Spring的AbstractRoutingDataSource解决多数据源的问题
# 详解利用Spring的AbstractRoutingDataSource解决多数据源的问题
# Spring(AbstractRoutingDataSource)实现动态数据源切换示例
# Spring多租户数据源管理 AbstractRoutingDataSour
# 自定义
# 是在
# 多个
# 绑定
# 完后
# 找不到
# 只需
# 它是
# 要在
# 要用
# 如果没有
# 要比
# 重写
# 可以使用
# 可以看出
# 时就
# 可以实现
# 涉及到
# 更强
# 到此
相关文章:
网站设计制作公司地址,网站建设比较好的公司都有哪些?
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
如何快速查询网址的建站时间与历史轨迹?
实现点击下箭头变上箭头来回切换的两种方法【推荐】
如何有效防御Web建站篡改攻击?
制作网站的基本流程,设计网站的软件是什么?
免费制作统计图的网站有哪些,如何看待现如今年轻人买房难的情况?
nginx修改上传文件大小限制的方法
大同网页,大同瑞慈医院官网?
文字头像制作网站推荐软件,醒图能自动配文字吗?
如何快速搭建高效香港服务器网站?
,如何利用word制作宣传手册?
网站制作中优化长尾关键字挖掘的技巧,建一个视频网站需要多少钱?
英语简历制作免费网站推荐,如何将简历翻译成英文?
如何快速生成高效建站系统源代码?
如何通过VPS建站无需域名直接访问?
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
,有什么在线背英语单词效率比较高的网站?
常州自助建站费用包含哪些项目?
5种Android数据存储方式汇总
php8.4新语法match怎么用_php8.4match表达式替代switch【方法】
,sp开头的版面叫什么?
高防服务器:AI智能防御DDoS攻击与数据安全保障
详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)
建站之星CMS五站合一模板配置与SEO优化指南
如何高效利用亚马逊云主机搭建企业网站?
建站之星如何一键生成手机站?
自助网站制作软件,个人如何自助建网站?
如何在云主机快速搭建网站站点?
宝盒自助建站智能生成技巧:SEO优化与关键词设置指南
整人网站在线制作软件,整蛊网站退不出去必须要打我是白痴才能出去?
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
建站之星安装失败:服务器环境不兼容?
武清网站制作公司,天津武清个人营业执照注销查询系统网站?
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
公司网站制作需要多少钱,找人做公司网站需要多少钱?
手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?
网站制作需要会哪些技术,建立一个网站要花费多少?
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
linux top下的 minerd 木马清除方法
如何在Golang中使用encoding/gob序列化对象_存储和传输数据
南平网站制作公司,2025年南平市事业单位报名时间?
香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南
制作网站公司那家好,网络公司是做什么的?
网站制作多少钱一个,建一个论坛网站大约需要多少钱?
高端建站三要素:定制模板、企业官网与响应式设计优化
外贸公司网站制作哪家好,maersk船公司官网?
相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?
岳西云建站教程与模板下载_一站式快速建站系统操作指南
网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?
*请认真填写需求信息,我们会在24小时内与您取得联系。