当前位置:
首页
文章
前端
详情

SASS平台根据用户机构动态切换数据库连接的datasource

背景:作为sass平台,有若干机构作为系统的租户存在,用户的创建需要绑定到唯一的机构下面,机构有机构简称,设计为,根据不同的机构下的用户设立独立的数据库,平台系统根据用户所在的机构去连接不同数据库进行业务操作

1.创建注解类

@Target(ElementType.METHOD,ElementType.Type)
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Ducumented
public @interface DBChangeAnno{
   String value() default "master";
}

2.在数据库连接配置文件中设置多个数据源,并且数据源的名称根据机构简称设置

xx.jdbc.datasource.names=master,czbk
xx.jdbc.datasource.master.jdbcurl=jdbc:mysql://192.1.1.1:3306/db1
xx.jdbc.datasource.master.username=db1
xx.jdbc.datasource.master.passowrd=123456
xx.jdbc.datasource.master.driverClassName=com.mysql.jdbc.Driver
....
xx.jdbc.datasource.czbk.jdbcurl=jdbc:mysql://192.1.1.2:3306/db2
xx.jdbc.datasource.czbk.username=db2
xx.jdbc.datasource.czbk.passowrd=123456
xx.jdbc.datasource.czbk.driverClassName=com.mysql.jdbc.Driver

3.创建切面类等类,对所有标注了注解的方法执行前进行拦截并且切换数据源

//编写切面类,对注解方法进行前置增强,改变当前线程的数据库连接
@Slf4j
@Aspect
@Order(-1)
@ConditionalOnProperty(value = {xx.jdbc.datasource.names})
@Component
public class DBChangeInterceptor {
   @Autowired
   private XXXmapper mapper;//查询用户所在机构简称mapper
   
   @Before("@annotation(dbChangeAnno)")
   //这里也可加@within(dbChangeAnno)就会在类层面的所有方法进行切面,但是不能写在一起,比如@Before("@annotation(dbChangeAnno)||@within(dbChangeAnno)")这样只会生效后面的
   public void switchDataSource(JoinPoint point,DBCHangeAnno anno){
      String sourceName = anno.value();//默认是连接master数据库的,因为用户及机构信息是在master数据库进行维护
      if(SessionUtils.getSession() != null){
         sourceName = SessionUtils.getSession().getOrgId();//在session中存储当前登录用户的机构代码主键
         OrgInfo orgi = mapper.sekectByPrimaryKey(sourceName);
         sourceName = orgi.getShortName();
      }
      if(XXDataSourceContextHolder.containDataSourceName(sourceName)){
         XXDataSourceContextHolder.setCurrentDataSourceName(sourceName);
      }
   }

   //调用结束后还原线程的连接数据库到默认库中去
   @After("@annotation(dbChangeAnno)")
   public void restoreDataSource(JoinPoint point,DBCHangeAnno anno){
      XXDataSourceContextHolder.clearCurrentDataSourceName();
   }
}

//多数据源上下文类
public final class XXDataSourceContextHolder{
   //数据源标识保存在线程变量中,避免多线程操作数据源时互相干扰。
   private static final ThreadLocal<String> CONTEXT_HOLDER = new ThreadLocal<String>();
   private static final Set<String> NAME_SET = new CopyOnWriteArraySet<String>();
   private static String defaultDataSourceName = null;//程序启动时将xx.jdbc.datasource.names的第一个名称作为默认数据库

   private XXDataSourceContextHolder(){
   }

   public static void setCurrentDataSourceName(String name){
      CONTEXT_HOLDER.set(name);
   }

   public static String getCurrentDataSourceName(){
      String name = CONTEXT_HOLDER.get();
      if(name == null){
         name = defaultDataSourceName;
      }
      return name;
   }

   public static void clearCurrentDataSourceName(){
      CONTEXT_HOLDER.remove();
   }

   public static boolean containDataSourceName(String name){
      return NAME_SET.contains(name);
   }

   public static void addDataSourceName(String name){
      NAME_SET.add(name);
   }

   public static void setDefaultDataSourceName(String name){
      XXDataSourceContextHolder.defaultDataSourceName = name;
   }
}

//在数据源的配置时使用该数据源配置
@Slf4j
public class XXRoutingDataSource extends AbstractRoutingDataSource{
   @Override
   protected Object determinCurrentLookupKey(){
      String cuurentDSName = XXDataSourceContextHolder.getCurrentDataSourceName();
      if(currentDSName == null || "".equals(cuurentDSName)){
         cuurentDSName = XXDataSourceContextHolder.getDefaultDataSourceName();
      }
      return cuurentDSName;
   }
}

//在Controller类中需要切换数据库的方法增加注解
@RequestMapping("/a/b/")
public class XXController{
   @RequestMapping(value = "list")
   @DBChangeAnno //增加此注解表示该业务需根据用户所在机构切换不同数据库连接
   public Obj list(){
      return new Object;//具体业务逻辑编写
   }
}

//参考文章:
https://www.jianshu.com/p/6b203f4926d5
https://www.cnblogs.com/haha12/p/10613549.html

免责申明:本站发布的内容(图片、视频和文字)以转载和分享为主,文章观点不代表本站立场,如涉及侵权请联系站长邮箱:xbc-online@qq.com进行反馈,一经查实,将立刻删除涉嫌侵权内容。