全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

使用Spring的AbstractRoutingDataSource实现多数据源切换示例

最近因为项目需要在做两个项目间数据同步的需求,具体是项目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  # 自定义  # 是在  # 多个  # 绑定  # 完后  # 找不到  # 只需  # 它是  # 要在  # 要用  # 如果没有  # 要比  # 重写  # 可以使用  # 可以看出  # 时就  # 可以实现  # 涉及到  # 更强  # 到此 


相关文章: 实现点击下箭头变上箭头来回切换的两种方法【推荐】  教程网站设计制作软件,怎么创建自己的一个网站?  如何通过VPS建站实现广告与增值服务盈利?  潍坊网站制作公司有哪些,潍坊哪家招聘网站好?  教学网站制作软件,学习*后期制作的网站有哪些?  建站之星与建站宝盒如何选择最佳方案?  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  广州网站设计制作一条龙,广州巨网网络科技有限公司是干什么的?  建站主机核心功能解析:服务器选择与网站搭建流程指南  网页设计与网站制作内容,怎样注册网站?  如何快速登录WAP自助建站平台?  大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?  宝塔面板创建网站无法访问?如何快速排查修复?  如何用腾讯建站主机快速创建免费网站?  动图在线制作网站有哪些,滑动动图图集怎么做?  如何配置支付宝与微信支付功能?  如何快速生成高效建站系统源代码?  如何批量查询域名的建站时间记录?  如何在IIS管理器中快速创建并配置网站?  如何在万网ECS上快速搭建专属网站?  常州企业建站如何选择最佳模板?  如何在云服务器上快速搭建个人网站?  建站之星如何快速更换网站模板?  单页制作网站有哪些,朋友给我发了一个单页网站,我应该怎么修改才能把他变成自己的呢,请求高手指点迷津?  如何在服务器上三步完成建站并提升流量?  小程序网站制作需要准备什么资料,如何制作小程序?  唐山网站制作公司有哪些,唐山找工作哪个网站最靠谱?  Swift开发中switch语句值绑定模式  Avalonia如何实现跨窗口通信 Avalonia窗口间数据传递  山东云建站价格为何差异显著?  专业网站制作企业网站,如何制作一个企业网站,建设网站的基本步骤有哪些?  ,石家庄四十八中学官网?  建站之星如何防范黑客攻击与数据泄露?  如何在阿里云虚拟服务器快速搭建网站?  如何在景安云服务器上绑定域名并配置虚拟主机?  网站视频怎么制作,哪个网站可以免费收看好莱坞经典大片?  C#如何在一个XML文件中查找并替换文本内容  高配服务器限时抢购:企业级配置与回收服务一站式优惠方案  小说建站VPS选用指南:性能对比、配置优化与建站方案解析  如何快速搭建安全的FTP站点?  制作网页的网站有哪些,电脑上怎么做网页?  如何快速使用云服务器搭建个人网站?  自助网站制作软件,个人如何自助建网站?  如何在Windows虚拟主机上快速搭建网站?  如何制作一个表白网站视频,关于勇敢表白的小标题?  为什么Go需要go mod文件_Go go mod文件作用说明  建站之星如何开启自定义404页面避免用户流失?  如何注册花生壳免费域名并搭建个人网站?  如何通过NAT技术实现内网高效建站?  如何在Golang中引入测试模块_Golang测试包导入与使用实践 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。