每一个请求与其他的用户是面对不同的数据库,这就需要用到动态数据源切换,来满足不同数据库、不同数据表(不同数据源)的灵活调用。
动态数据源切换
满足mysql、oracle等主流数据库进行动态数据源切换。
使用方法
- 在 WEB-INF 目录下建立 lib 目录,并把jar包复制到 lib 目录;
- 在maven配置文件,即 pom.xml 中,需要包含以下依赖:
org.aspectj
aspectjweaver
1.8.13
org.springframework
spring-context
5.0.4.RELEASE
org.springframework
spring-web
5.0.4.RELEASE
javax.servlet
servlet-api
2.5
provided
dynamicds
dynamicds
1.0
system
${project.basedir}/src/main/webapp/WEB-INF/lib/dynamicds.jar
注册项目的数据源(这是必须要做的)
继承DynamicDataSourceRegister类,并完成相关的方法重写。
import com.ycj.fastframe.dynamicds.aop.DynamicDataSourceRegister;
import com.ycj.fastframe.dynamicds.entity.DataBaseInfo;
import org.springframework.core.env.Environment;
import java.util.List;
/**
* @author: Frank
* @email 1320259466@qq.com
* @date: 2019/11/17
* @time: 12:57
* @fuction: about the role of class.
*/
public class MyDataSource extends DynamicDataSourceRegister {
@Override
public DataBaseInfo initDefaultDataSource(Environment env) {
// 读取主数据源
DataBaseInfo dataBaseInfo=new DataBaseInfo();
dataBaseInfo.setDriver(env.getProperty("spring.datasource.driver"));
dataBaseInfo.setUrl(env.getProperty("spring.datasource.url"));
dataBaseInfo.setUsername(env.getProperty("spring.datasource.username"));
dataBaseInfo.setPassword(env.getProperty("spring.datasource.password"));
dataBaseInfo.setType("com.alibaba.druid.pool.DruidDataSource");
return dataBaseInfo;
}
@Override
public List initOtherDataSources(Environment env) {
return null;
}
}
设置数据源切换规则(这个可以四选一实现就行)
架构中采用了LRU算法的缓存策略,以适应高并发和高性能需求。这里举例其中一个实现策略,只需要继承BindDynamicDataSourceByUserIdentifier类,并完成相关的方法重写。
import com.ycj.fastframe.dynamicds.aop.BindDynamicDataSourceByUserIdentifier;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
/**
* @author: Frank
* @email 1320259466@qq.com
* @date: 2019/11/16
* @time: 14:00
* @fuction: about the role of class.
*/
@Aspect
@Order(-1)//保证在@Transactional之前执行
@Component
public class TestDynamicDataSource extends BindDynamicDataSourceByUserIdentifier {
@Override
@Pointcut("execution(public * com.ycj.fastframe.controller..*.*(..))")
public void dataSourceAspectPath() {
}
@Override
public void logPrint(String msg) {
System.out.println(msg);
}
@Override
public int initCacheSize() {
return 100;
}
@Override
public String setDataSouceForUser(String userUniqueIdentifier) {
if (userUniqueIdentifier.contains("1001")){
return "project1";
}else if (userUniqueIdentifier.contains("1002")){
return "project2";
}else {
return "dataSource";
}
}
@Override
public String getUserUniqueIdentifier(HttpServletRequest request, String requestArgs) {
return request.getSession().getId();
}
public static void main(String[] args) {
}
}
四种方法
动态数据源切换有四种实现方式:
一、通过注解,这个最简单
只需要继承BindDynamicDataSourceByAnnotation类,并在controller中的每个方法前面添加 @RegisterDataSource(name = “数据源名称”) 注解即可,如下
@Aspect
@Order(-1)//保证在@Transactional之前执行
@Component
public class TestDynamicDataSource extends BindDynamicDataSourceByAnnotation {
}
二、通过继承BindDynamicDataSourceByArgs类,来实现,步骤跟上面一样,都很简单。
三、通过继承BindDynamicDataSourceByRequest类,来实现,步骤跟上面一样,都很简单。
四、通过继承BindDynamicDataSourceByUserIdentifier类,来实现,步骤跟上面一样,都很简单。
原文链接:https://blog.csdn.net/u014374009/article/details/103108025