spring 多个数据库之间切换

多数据源问题很常见,例如读写分离数据库配置。

原来的项目出现了新需求,局方要求新增某服务器用以提供某代码,涉及到多数据源的问题。

研究成果如下:

1、首先配置多个datasource

[html] view plaincopy
  1. <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">  
  2.         <property name="driverClassName" value="net.sourceforge.jtds.jdbc.Driver">  
  3.         </property>  
  4.         <property name="url" value="jdbc:jtds:sqlserver://10.82.81.51:1433;databaseName=standards">  
  5.         </property>  
  6.         <property name="username" value="youguess"></property>  
  7.         <property name="password" value="youguess"></property>  
  8.     </bean>  
  9.     <bean id="dataSource2" class="org.apache.commons.dbcp.BasicDataSource">  
  10.         <property name="driverClassName" value="net.sourceforge.jtds.jdbc.Driver">  
  11.         </property>  
  12.         <property name="url" value="jdbc:jtds:sqlserver://10.82.81.52:1433;databaseName=standards">  
  13.         </property>  
  14.         <property name="username" value="youguess"></property>  
  15.         <property name="password" value="youguess"></property>  
  16. </bean>  


2、写一个DynamicDataSource类继承AbstractRoutingDataSource,并实现determineCurrentLookupKey方法

[java] view plaincopy
  1. package com.standard.core.util;  
  2. import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;  
  3. public class DynamicDataSource extends AbstractRoutingDataSource {  
  4.     @Override  
  5.     protected Object determineCurrentLookupKey() {  
  6.         return CustomerContextHolder.getCustomerType();  
  7.     }  
  8. }  



3、利用ThreadLocal解决线程安全问题

[java] view plaincopy
  1. package com.standard.core.util;  
  2. public class CustomerContextHolder {  
  3.     public static final String DATA_SOURCE_A = "dataSource";  
  4.     public static final String DATA_SOURCE_B = "dataSource2";  
  5.     private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>();  
  6.     public static void setCustomerType(String customerType) {  
  7.         contextHolder.set(customerType);  
  8.     }  
  9.     public static String getCustomerType() {  
  10.         return contextHolder.get();  
  11.     }  
  12.     public static void clearCustomerType() {  
  13.         contextHolder.remove();  
  14.     }  
  15. }  

4、数据源配置

[html] view plaincopy
  1. <bean id="dynamicDataSource" class="com.standard.core.util.DynamicDataSource" >  
  2.         <property name="targetDataSources">  
  3.             <map key-type="java.lang.String">  
  4.                 <entry value-ref="dataSource" key="dataSource"></entry>  
  5.                 <entry value-ref="dataSource2" key="dataSource2"></entry>  
  6.             </map>  
  7.         </property>  
  8.         <property name="defaultTargetDataSource" ref="dataSource" >  
  9.         </property>  
  10.     </bean>   

5、在DAOImpl代码中手动切换数据源

[java] view plaincopy
  1. CustomerContextHolder.setCustomerType(CustomerContextHolder.DATA_SOURCE_B);   

6、或者用AOP动态切换数据源

package datasource;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;

@Aspect
public class DynamicDataSourceAspect {
 @Pointcut("execution (public service.impl..*.*(..))")
 public void serviceExecution(){}
 
 @Before("serviceExecution()")
 public void setDynamicDataSource(JoinPoint jp) {
  for(Object o : jp.getArgs()) {
   //处理具体的逻辑 ,根据具体的境况CustomerContextHolder.setCustomerType()选取DataSource
  }
 }
}


搞定!


posted @   ppjj  阅读(805)  评论(0编辑  收藏  举报
编辑推荐:
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 三行代码完成国际化适配,妙~啊~
· .NET Core 中如何实现缓存的预热?
点击右上角即可分享
微信分享提示