iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换
  • 537
分享到

SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换

2023-06-30 16:06:16 537人浏览 独家记忆
摘要

本文小编为大家详细介绍“SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换”,内容详细,步骤清晰,细节处理妥当,希望这篇“springBoot基于AbstractRoutingDataSour

本文小编为大家详细介绍“SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换”,内容详细,步骤清晰,细节处理妥当,希望这篇“springBoot基于AbstractRoutingDataSource如何实现多数据源动态切换”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

一、场景

在生产业务中,有一些任务执行了耗时较长的查询操作,在实时性要求不高的时候,我们希望将这些查询sql分离出来,去从库查询,以减少应用对主数据库的压力。

一种方案是在配置文件中配置多个数据源,然后通过配置类来获取数据源以及mapper相关的扫描配置,不同的数据源配置不佟的mapper扫描位置,然后需要哪一个数据源就注入哪一个mapper接口即可,这种方法比较简单。特征是通过mapper扫描位置区分数据源。

第二种方案是配置一个默认使用的数据源,然后定义多个其他的数据源,使用aop形成注解式选择数据源。此种方案实现的核心是对AbstractRoutingDataSource 类的继承。这是本文的重点。

二、原理

AbstractRoutingDataSource的多数据源动态切换的核心逻辑是:在程序运行时,把数据源数据源通过 AbstractRoutingDataSource 动态织入到程序中,灵活的进行数据源切换。
基于AbstractRoutingDataSource的多数据源动态切换,可以实现读写分离。逻辑如下:

    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;    }    @Nullable    protected abstract Object determineCurrentLookupKey();

通过实现抽象方法determineCurrentLookupKey指定需要切换的数据源

三、代码示例

示例中主要依赖 

com.alibaba.druid;tk.mybatis

定义一个类用于关联数据源。通过 TheadLocal 来保存每个线程选择哪个数据源的标志(key)

@Slf4jpublic class DynamicDataSourceContextHolder {      private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>();    public static List<String> dataSourceIds = new ArrayList<String>();     public static void setDataSourceType(String dataSourceType) {        log.info("设置当前数据源为{}",dataSourceType);        contextHolder.set(dataSourceType);    }     public static String getDataSourceType() {        return contextHolder.get() ;    }     public static void clearDataSourceType() {        contextHolder.remove();    }     public static boolean containsDataSource(String dataSourceId){        log.info("list = {},dataId={}", JSON.tojsON(dataSourceIds),dataSourceId);        return dataSourceIds.contains(dataSourceId);    }}

继承

AbstractRoutingDataSource

public class DynamicDataSource  extends AbstractRoutingDataSource {     @Override    protected Object determineCurrentLookupKey() {         return  DynamicDataSourceContextHolder.getDataSourceType();    }}

配置主数据库master 与从数据库slave(略)。数据源配置可以从简

@Configuration@tk.mybatis.spring.annotation.MapperScan(value = {"com.server.dal.dao"})@ConditionalOnProperty(name = "java.druid.datasource.master.url")public class JavaDruidDataSourceConfiguration {     private static final Logger logger = LoggerFactory.getLogger(JavaDruidDataSourceConfiguration.class);     @Resource    private JavaDruidDataSourceProperties druidDataSourceProperties;    @Primary    @Bean(name = "masterDataSource", initMethod = "init", destroyMethod = "close")    @ConditionalOnMissingBean(name = "masterDataSource")    public DruidDataSource javaReadDruidDataSource() {         DruidDataSource result = new DruidDataSource();         try {//            result.setName(druidDataSourceProperties.getName());            result.setUrl(druidDataSourceProperties.getUrl());            result.setUsername(druidDataSourceProperties.getUsername());            result.setPassword(druidDataSourceProperties.getPassword());            result.setConnectionProperties(                    "config.decrypt=false;config.decrypt.key=" + druidDataSourceProperties.getPwdPublicKey());            result.setFilters("config");            result.setMaxActive(druidDataSourceProperties.getMaxActive());            result.setInitialSize(druidDataSourceProperties.getInitialSize());            result.setMaxWait(druidDataSourceProperties.getMaxWait());            result.setMinIdle(druidDataSourceProperties.getMinIdle());            result.setTimeBetweenEvictionRunsMillis(druidDataSourceProperties.getTimeBetweenEvictionRunsMillis());            result.setMinEvictableIdleTimeMillis(druidDataSourceProperties.getMinEvictableIdleTimeMillis());            result.setValidationQuery(druidDataSourceProperties.getValidationQuery());            result.setTestWhileIdle(druidDataSourceProperties.isTestWhileIdle());            result.setTestOnBorrow(druidDataSourceProperties.isTestOnBorrow());            result.setTestOnReturn(druidDataSourceProperties.isTestOnReturn());            result.setPoolPreparedStatements(druidDataSourceProperties.isPoolPreparedStatements());            result.setMaxOpenPreparedStatements(druidDataSourceProperties.getMaxOpenPreparedStatements());             if (druidDataSourceProperties.isEnableMonitor()) {                StatFilter filter = new StatFilter();                filter.setLogSlowSql(druidDataSourceProperties.isLogSlowSql());                filter.setMergeSql(druidDataSourceProperties.isMergeSql());                filter.setSlowSqlMillis(druidDataSourceProperties.getSlowSqlMillis());                List<Filter> list = new ArrayList<>();                list.add(filter);                result.setProxyFilters(list);            }         } catch (Exception e) {             logger.error("数据源加载失败:", e);         } finally {            result.close();        }          return result;    }}

注意主从数据库的bean name

配置DynamicDataSource 

  • targetDataSources 存放数据源的k-v对

  • defaultTargetDataSource 存放默认数据源

配置事务管理器和SqlSessionFactoryBean

@Configurationpublic class DynamicDataSourceConfig {    private static final String MAPPER_LOCATION = "classpath*:sqlmap/dao/*Mapper.xml";     @Bean(name = "dynamicDataSource")    public DynamicDataSource dynamicDataSource(@Qualifier("masterDataSource") DruidDataSource masterDataSource,                                               @Qualifier("slaveDataSource") DruidDataSource slaveDataSource) {        Map<Object, Object> targetDataSource = new HashMap<>();        DynamicDataSourceContextHolder.dataSourceIds.add("masterDataSource");        targetDataSource.put("masterDataSource", masterDataSource);        DynamicDataSourceContextHolder.dataSourceIds.add("slaveDataSource");        targetDataSource.put("slaveDataSource", slaveDataSource);        DynamicDataSource dataSource = new DynamicDataSource();        dataSource.setTargetDataSources(targetDataSource);        dataSource.setDefaultTargetDataSource(masterDataSource);        return dataSource;    }     @Primary    @Bean(name = "javaTransactionManager")    @ConditionalOnMissingBean(name = "javaTransactionManager")    public DataSourceTransactionManager transactionManager(@Qualifier("dynamicDataSource") DynamicDataSource druidDataSource) {        return new DataSourceTransactionManager(druidDataSource);    }     @Bean(name = "sqlSessionFactoryBean")    public SqlSessionFactoryBean myGetSqlSessionFactory(@Qualifier("dynamicDataSource") DynamicDataSource  dataSource) {        SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();        ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();        try {            sqlSessionFactoryBean.setMapperLocations(resolver.getResources(MAPPER_LOCATION));        } catch (IOException e) {            e.printStackTrace();        }        sqlSessionFactoryBean.setDataSource(dataSource);        return sqlSessionFactoryBean;    }}

定义一个注解用于指定数据源

@Target({ ElementType.METHOD, ElementType.TYPE })@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface TargetDataSource {    String value();}

切面的业务逻辑。注意指定order,以确保在开启事务之前执行 。

@Aspect@Slf4j@Order(-1)@Componentpublic class DataSourceAop {     @Before("@annotation(targetDataSource)")    public void changeDataSource(JoinPoint point, TargetDataSource targetDataSource) {        String dsId = targetDataSource.value();        if (!DynamicDataSourceContextHolder.containsDataSource(dsId)) {            log.error("数据源[{}]不存在,使用默认数据源 > {}" + targetDataSource.value() + point.getSignature());        } else {            log.info("UseDataSource : {} > {}" + targetDataSource.value() + point.getSignature());            DynamicDataSourceContextHolder.setDataSourceType(targetDataSource.value());         }    }     @After("@annotation(targetDataSource)")    public void restoreDataSource(JoinPoint point, TargetDataSource targetDataSource) {        log.info("RevertDataSource : {} > {}"+targetDataSource.value()+point.getSignature());        DynamicDataSourceContextHolder.clearDataSourceType();    }}

以上略去了pom.xml和application.yml

使用示例

    @Resource    private ShopBillDOMapper shopBillDOMapper; //使用默认数据源    public ShopBillBO queryTestData(Integer id){         return shopBillDOMapper.getByShopBillId(id);    } //切换到指定的数据源    @TargetDataSource("slaveDataSource")    public ShopBill queryTestData2(Integer id){        return shopBillDOMapper.getByShopBillId(id);    }

如果返回不同的结果就成功了!

读到这里,这篇“SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网精选频道。

--结束END--

本文标题: SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换

本文链接: https://www.lsjlt.com/news/330188.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 如何在 Golang 中替换正则表达式匹配的文本?
    在 go 中,可使用 regexp.replaceall 函数替换符合正则表达式的文本,该函数需要三个参数:待替换字符串、匹配模式和替换文本。例如,将字符串中 "fox" 替换为 "do...
    99+
    2024-05-14
    golang 正则表达式
  • 如何在 Golang 中测试随机数生成器的准确性?
    在 go 中测试随机数生成器准确性的步骤包括:生成大量随机数并计算每个范围内的出现次数,以确保均匀分布。针对指定均值和标准差计算每个范围内的出现次数,以确保正态分布。 如何在 Gola...
    99+
    2024-05-14
    golang 随机数
  • 面向对象设计原则在C++中的体现
    c++++ 体现了 oop 原则,包括:封装:使用类将数据和方法封装在对象中。继承:允许派生类从基类继承数据和行为。多态:允许对象的行为根据其类型而改变,通过虚函数实现。 面向对象设计...
    99+
    2024-05-14
    c++ 面向对象
  • c语言怎么区分小数和整数
    c 语言区分小数和整数的方法有:数据类型不同:小数类型(float、double)包含小数点,整数类型(int)不包含。printf() 函数中使用不同格式化字符串:小数用 %f,整数用...
    99+
    2024-05-14
    c语言
  • 设计模式在C++ 中的可复用性和可扩展性
    在 c++++ 中,设计模式通过提供经过验证的解决方案来提高可复用性和可扩展性。可复用性允许重复使用代码,例如 factory method 模式,它支持创建不同的产品而不影响具体类。可...
    99+
    2024-05-14
    c++ 设计模式 高可扩展性
  • C++语法中函数模板的灵活运用
    C++ 语法中函数模板的灵活运用 函数模板是 C++ 中的一项强大功能,允许您创建可用于不同数据类型的一组代码。这可以提高代码的可重用性,并使您能够编写更通用、更可维护的代码。 语法 ...
    99+
    2024-05-14
    c++语法 函数模板 c++
  • c语言怎么计算字符串长度和宽度
    在 c 语言中,计算字符串长度和宽度的函数分别为:strlen() 函数用于计算字符串长度,不包括终止符 '\0'。strwidth() 函数用于计算字符串在终端中的宽度,返回显示像素数...
    99+
    2024-05-14
    c语言
  • 如何用 Golang 正则匹配多个单词或字符串?
    golang 正则表达式使用管道符 | 来匹配多个单词或字符串,将各个选项作为逻辑 or 表达式分隔开来。例如:匹配 "fox" 或 "dog":fox|dog匹配 "quick"、"b...
    99+
    2024-05-14
    golang 正则 python
  • c语言怎么跳出多层循环
    在 c 语言中,可以使用嵌套的 break 语句跳出多层循环。对于每个要跳出的循环层,都需要一个单独的 break 语句。例如:使用一个 break 语句跳出内层循环再使用一个 brea...
    99+
    2024-05-14
    c语言
  • c语言怎么注释成中文
    c语言中文注释提供两种方式:行内注释(以"//"开头)和块注释(以"/"开头并以"/"结尾)。最佳实践包括:使用简明扼要的语言,在函数和类开头处添加块注释,在关键部分添加行内注释,保持注...
    99+
    2024-05-14
    c语言
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作