当前位置: 首页 > news >正文

网络规划设计师薪资一键优化

网络规划设计师薪资,一键优化,wordpress 完整主题下载,wordpress 加上wwwSpringBoot整合多数据源,动态添加新数据源并切换 1.需求2.创建数据源配置类3.切换数据源4.切换数据源管理类5.使用案例5.AOP切面拦截 1.需求 低代码服务需要给多套系统进行功能配置,要求表结构必须生成在对应系统的数据库中,所以表结构的生成…

SpringBoot整合多数据源,动态添加新数据源并切换

    • 1.需求
    • 2.创建数据源配置类
    • 3.切换数据源
    • 4.切换数据源管理类
    • 5.使用案例
    • 5.AOP切面拦截

1.需求

低代码服务需要给多套系统进行功能配置,要求表结构必须生成在对应系统的数据库中,所以表结构的生成需要动态的获取目标系统的数据库信息,切换当前数据源到目标数据库,然后再生成。

2.创建数据源配置类

package com.tyq.datasource.config.datasource;import com.alibaba.druid.pool.DruidDataSource;
import com.tyq.datasource.config.Properties;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;import javax.annotation.Resource;
import javax.sql.DataSource;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;/*** 向Spring容器中注入DruidConfiguration** @author 谭永强*/
@Configuration
@EnableAutoConfiguration(exclude = {DataSourceAutoConfiguration.class})
public class DruidConfiguration {@Resourceprivate Properties properties;/*** 数据源(默认)*/public DataSource dataSourceToDefault() {DruidDataSource datasource = new DruidDataSource();try {datasource.setQueryTimeout(0);datasource.setUrl("jdbc:oracle:thin:@192.168.0.30:1521/test2");datasource.setUsername("SCPS");datasource.setPassword("yldtscps");} catch (SQLException e) {e.printStackTrace();}return datasource;}@Bean@Primarypublic DataSource dataSource() {Map<Object, Object> dataSourceMap = new HashMap<>(2);dataSourceMap.put("default", dataSourceToDefault());DynamicDatasource dynamicDatasource = new DynamicDatasource();// 注入目标数据源,如果有多个数据源,直接加入map即可dynamicDatasource.setTargetDataSources(dataSourceMap);// 注入默认数据源dynamicDatasource.setDefaultTargetDataSource(dataSourceToDefault());return dynamicDatasource;}
}

3.切换数据源

SpringBoot动态切换数据源主要依靠AbstractRoutingDataSource类,这个抽象类中有一个属性为targetDataSources。该属性为Map结构,所有需要切换的数据源都存放在其中,根据指定的KEY进行切换。
在AbstractRoutingDataSource源码中,获取数据库连接是通过this.determineTargetDataSource().getConnection()去获取的,而this.determineTargetDataSource()方法中的DataSource则是通过determineCurrentLookupKey()方法返回的key值去前面的dynamicDatasource.setTargetDataSources(dataSourceMap);设置进去的map中查找的。

所以我们需要重写determineCurrentLookupKey()方法,如下:

package com.tyq.datasource.config.datasource;import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;/*** 动态数据源** @author 谭永强* @date 2023-08-03*/
public class DynamicDatasource extends AbstractRoutingDataSource {@Overrideprotected Object determineCurrentLookupKey() {return DynamicDatasourceHolder.getDataSource();}
}

4.切换数据源管理类

数据源属于公共资源,考虑到多线程的情况下,我们将数据源存储在【ThreadLocal】中,保证线程隔离。

package com.tyq.datasource.config.datasource;import java.util.Queue;
import java.util.concurrent.ArrayBlockingQueue;/*** 数据源切换管理** @author 谭永强* @date 2023-08-03*/
public class DynamicDatasourceHolder {/*** 保存数据源的映射*/private final static ThreadLocal<String> CONTEXT_HOLDER = new ThreadLocal<>();public static String getDataSource() {return CONTEXT_HOLDER.get();}public static void setDataSource(String dataSourceKey) {CONTEXT_HOLDER.set(dataSourceKey);}public static void removeDataSource() {CONTEXT_HOLDER.remove();}
}

5.使用案例

在Controller中通过DynamicDatasourceHolder指定当前需要使用哪个数据源,具体使用如下:

@RequestMapping("findById")
public User findById(String userId) {//指定数据源DynamicDatasourceHolder.setDataSource("default");if (ObjectUtils.isEmpty(userId)) {throw new ParamValidateException("userId不能为空");}User user = userService.findById(userId);//移除当前数据源DynamicDatasourceHolder.removeDataSource();return user;
}

上述案例中已经实现了如果动态切换数据源的过程,但是在实际开发中还是太繁琐,比如每个接口都必须添加相关切换数据源的代码,对代码的侵入性太高,下面我们通过AOP的形式去优化切换数据源的过程。

5.AOP切面拦截

通过AOP切面对方法的前置和后置做切换数据源的操作,这样就降低了与业务代码耦合、

<!--AOP-->
<dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.9.6</version>
</dependency>

注意:如果项目中无法导入@Aspect,则需要添加上述依赖。

package com.tyq.datasource.aop;import com.alibaba.druid.pool.DruidDataSource;
import com.tyq.datasource.config.Properties;
import com.tyq.datasource.config.datasource.DynamicDatasource;
import com.tyq.datasource.config.datasource.DynamicDatasourceHolder;
import com.tyq.datasource.config.exception.ParamValidateException;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;
import org.springframework.util.ObjectUtils;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.sql.DataSource;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;/*** 动态数据源切换AOP** @author 谭永强* @date 2023-08-03*/
@Aspect
@Component
public class DynamicDatasourceAop {@Autowiredprotected ApplicationContext applicationContext;@Resourceprivate Properties properties;/*** 定义切点*/@Pointcut("execution (* com.tyq.datasource.controller.*.*(..))")public void pointcut() {}/*** 前置处理*/@Before(value = "pointcut()")public void beforeAdvice() {// 接收到请求,记录请求内容ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();if (ObjectUtils.isEmpty(attributes)) {return;}HttpServletRequest request = attributes.getRequest();String dataSourceKey = request.getParameter("dataSourceKey");if (ObjectUtils.isEmpty(dataSourceKey)) {throw new ParamValidateException("dataSourceKey不能为空!");}//获取当前动态数据源DynamicDatasource dynamicDatasource = applicationContext.getBean(DynamicDatasource.class);//所有已连接的数据源集合Map<Object, DataSource> resolvedDataSources = dynamicDatasource.getResolvedDataSources();if (ObjectUtils.isEmpty(resolvedDataSources.get(dataSourceKey))) {//此处为模拟到数据库中查询数据源信息的过程,查询到数据源信息后,创建数据源并添加到动态数据源管理中//动态加入新的数据源DataSource dataSource = getDataSource();Map<Object, Object> dataSourceMap = new HashMap<>(resolvedDataSources.size() + 1);dataSourceMap.putAll(resolvedDataSources);dataSourceMap.put("three", dataSource);dynamicDatasource.setTargetDataSources(dataSourceMap);}//动态切换数据源DynamicDatasourceHolder.setDataSource(dataSourceKey);}/*** 后置处理*/@AfterReturning(value = "pointcut()")public void afterAdvice() {// 接收到请求,记录请求内容ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();if (ObjectUtils.isEmpty(attributes)) {return;}HttpServletRequest request = attributes.getRequest();String dataSourceKey = request.getParameter("dataSourceKey");if (ObjectUtils.isEmpty(dataSourceKey)) {throw new ParamValidateException("dataSourceKey不能为空!");}System.out.println("后置处理:" + dataSourceKey);//删除当前数据源DynamicDatasourceHolder.removeDataSource();}/*** 模拟成功查询到数据源并创建DataSource** @return 数据源*/public DataSource getDataSource() {DruidDataSource datasource = new DruidDataSource();try {datasource.setQueryTimeout(0);datasource.setUrl("jdbc:oracle:thin:@192.168.0.59:1526/qstest");datasource.setUsername("SCPS");datasource.setPassword("yldtscps");} catch (SQLException e) {e.printStackTrace();}return datasource;}
}
http://www.mmbaike.com/news/69167.html

相关文章:

  • 湛江seo建站搜索引擎优化的特点
  • 自己建网站数据怎么做新浪新闻疫情
  • 公司装修有甲醛味要上班怎么办关键词优化外包服务
  • 静乐县城乡建设局网站流量网站
  • 论文网站建设的参考文献福建优化seo
  • 网站建设 website微信营销平台系统
  • 做赌博网站赚中国知名网站排行榜
  • 做网站一般要了解哪些自媒体平台app
  • 中英文网站源码今天国际新闻最新消息
  • 天津市建设委员会 网站排名轻松seo 网站推广
  • 网站一屏做多大济南网站建设制作
  • 帮别人做买家秀的网站推广怎么推
  • 如何制作微信公众号微商城国外网站seo
  • 南京机械加工网seo基础篇
  • 有哪些外国网站国内可以登录的门户网站建站系统
  • 全球速卖通注册条件沧州网站优化
  • 商业网站推广自媒体135网站免费下载安装
  • 企业大型网站开发网站模板设计nba最新交易动态
  • 贺州招聘网站建设百度投诉中心24人工 客服电话
  • wordpress 独立 分类温州seo公司
  • 做网站专家seo上首页排名
  • 湖南建设网站推广软文模板
  • 如何做企业网站php餐饮营销引流都有什么方法
  • 布吉商城网站建设基本流程推广营销平台
  • 现在还有用dw做网站百度网址大全官方下载
  • 智能家居产品设计案例windows优化大师有毒吗
  • 做pc端大型网站 前端用百度查重入口
  • 学做招投标的网站有哪些湘潭网站设计外包服务
  • 网站被刷怎么办贵州萝岗seo整站优化
  • 山东省安全双体系建设网站地址营销方案策划书