springboot使用DynamicDataSource动态切换数据源的实现过程

 更新时间:2023年08月07日 10:00:44   作者:菜鸟的IT学习之路  
这篇文章主要给大家介绍了关于springboot使用DynamicDataSource动态切换数据源的实现过程,Spring Boot应用中可以配置多个数据源,并根据注解灵活指定当前使用的数据源,需要的朋友可以参考下

DynamicDataSource是一个数据源路由器,可以根据上下文动态选择数据源。可以在每个请求或线程中将数据源设置为当前需要使用的数据.

1. 创建一个DynamicDataSource类来实现数据源路由逻辑

创建一个DynamicDataSource类,它继承自AbstractRoutingDataSource。在该类中重写**determineCurrentLookupKey()**方法,该方法返回一个字符串,用于指示当前要使用哪个数据源

public class DynamicDataSource extends AbstractRoutingDataSource {
    @Override
    protected Object determineCurrentLookupKey() {
        return DynamicDataSourceContextHolder.getDataSourceName();
    }
    public DynamicDataSource(DataSource defaultTargetDataSource, Map<Object, Object> targetDataSources) {
        super.setDefaultTargetDataSource(defaultTargetDataSource);
        super.setTargetDataSources(targetDataSources);
        super.afterPropertiesSet();
    }
}

2. 创建DynamicDataSourceContextHolder线程安全类

在多线程环境下,如果多个线程同时访问同一个方法,并且每个线程要使用不同的数据源,那么就需要对数据源进行动态切换。如果在方法中使用一个共享的变量来存储当前要使用的数据源名称,那么就会存在线程安全问题,可能会导致不同线程之间的数据源切换混乱,或者数据源切换不成功的情况发生。

为了解决这个问题,可以使用ThreadLocal来存储当前要使用的数据源名称。ThreadLocal是一种线程本地存储机制,它可以为每个线程提供一个独立的变量副本,使得每个线程都可以独立地操作自己的变量副本,而不会影响其他线程的变量副本。

在使用DynamicDataSource进行数据源切换时,每个线程都可以通过ThreadLocal来独立地设置和获取当前要使用的数据源名称,避免了多个线程之间数据源切换的混乱和不成功的情况。同时,在方法执行完毕后,使用ThreadLocal也可以避免内存泄漏问题。使用ThreadLocal来存储当前要使用的数据源名称。**setDataSource()**方法用于设置当前要使用的数据源名称,**getDataSource()**方法用于获取当前要使用的数据源名称,**clearDataSource()**方法用于清除当前要使用的数据源名称

public class DynamicDataSourceContextHolder {
    private static final ThreadLocal<String> contextHolder = new ThreadLocal<>();
    /**
     * 设置当前数据源的名称
     */
    public static void setDataSourceName(String dataSourceName) {
        contextHolder.set(dataSourceName);
    }
    /**
     * 获取当前数据源的名称
     */
    public static String getDataSourceName() {
        return contextHolder.get();
    }
    /**
     * 清除当前数据源的名称
     */
    public static void clearDataSourceName() {
        contextHolder.remove();
    }
}

3.创建多数据源配置

通过yml配置,将多数据源获取到MutilDataSourceProperties

spring:
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://****:3306/emp_ts?nullNamePatternMatchesAll=true&tinyInt1isBit=false&useSSL=false&nullCatalogMeansCurrent=true&useUnicode=true&characterEncoding=UTF-8&serverTimezone=GMT%2B8
    username: root
    password: root
mutil-datasource:
  connection:
    - dbName: dataSource1
      dbDriver: com.mysql.cj.jdbc.Driver
      dbUrl: jdbc:mysql://****:3306/emp_ts_1?nullNamePatternMatchesAll=true&tinyInt1isBit=false&useSSL=false&nullCatalogMeansCurrent=true&useUnicode=true&characterEncoding=UTF-8&serverTimezone=GMT%2B8
      dbUsername: root
      dbPassword: root
public class DbConnection {
    private String dbName;
    private String dbDialect;
    private String dbDriver;
    private String dbUrl;
    private String dbUsername;
    private String dbPassword;
    public DbConnection() {
    }
    public String getDbName() {
        return this.dbName;
    }
    public String getDbDialect() {
        return this.dbDialect;
    }
    public String getDbDriver() {
        return this.dbDriver;
    }
    public String getDbUrl() {
        return this.dbUrl;
    }
    public String getDbUsername() {
        return this.dbUsername;
    }
    public String getDbPassword() {
        return this.dbPassword;
    }
    public void setDbName(String dbName) {
        this.dbName = dbName;
    }
    public void setDbDialect(String dbDialect) {
        this.dbDialect = dbDialect;
    }
    public void setDbDriver(String dbDriver) {
        this.dbDriver = dbDriver;
    }
    public void setDbUrl(String dbUrl) {
        this.dbUrl = dbUrl;
    }
    public void setDbUsername(String dbUsername) {
        this.dbUsername = dbUsername;
    }
    public void setDbPassword(String dbPassword) {
        this.dbPassword = dbPassword;
    }
}
@Configuration
@ConfigurationProperties(
    prefix = "mutil-datasource"
)
public class MutilDataSourceProperties {
    private List<DbConnection> connection = new ArrayList();
    public MutilDataSourceProperties() {
    }
    public List<DbConnection> getConnection() {
        return this.connection;
    }
    public void setConnection(List<DbConnection> connection) {
        this.connection = connection;
    }
}

4.创建DataSourceConfig

在配置类中创建默认数据源及获取其他多数据源进行创建,默认数据源为spring.datasource下配置的数据源

@Configuration
public class DataSourceConfig {
    @Resource
    private MutilDataSourceProperties mutilDataSourceProperties;
    public DataSourceConfig() {
    }
    @Bean
    @ConfigurationProperties("spring.datasource")
    public DataSource hikariDataSource(DataSourceProperties properties) {
        return DataSourceBuilder.create(properties.getClassLoader()).driverClassName(properties.determineDriverClassName()).url(properties.determineUrl()).username(properties.determineUsername()).password(properties.determinePassword()).build();
    }
    @Bean
    @Primary
    public DynamicDataSource dynamicDataSource(DataSource hikariDataSource) {
        Map<Object, Object> targetDataSources = this.createTargetDataSource();
        return new DynamicDataSource(hikariDataSource, targetDataSources);
    }
     private Map<Object, Object> createTargetDataSource() {
        Map<Object, Object> targetDataSources = new HashMap();
        List<DbConnection> connections = this.mutilDataSourceProperties.getConnection();
        connections.forEach(e -> {
            DataSource dataSource = this.createDataSource(e);
            targetDataSources.put(e.getDbName(), dataSource);
        });
        return targetDataSources;
     }
     public DataSource createDataSource(DbConnection connection) {
        return DataSourceBuilder.create().driverClassName(connection.getDbDriver()).url(connection.getDbUrl())
                .username(connection.getDbName()).password(connection.getDbPassword()).build();
     }
}

5. 使用

@Service
public class UserServiceImpl implements UserService {
    @Autowired
    private UserMapper userMapper;
    @Override
    public void addUser(User user) {
        DynamicDataSource.setDataSource("dataSource1");
        userMapper.insert(user);
    }
    @Override
    public void updateUser(User user) {
        DynamicDataSource.setDataSource("dataSource1");
        userMapper.updateById(user);
    }
}

6.优缺点

优点:

  • 简单易用:使用 DynamicDataSource 和 ThreadLocal 进行动态数据源切换,配置相对简单,易于上手。
  • 可扩展性强:通过继承 AbstractRoutingDataSource 类,可以实现自定义的数据源路由策略。并且,由于采用了抽象类,扩展性也比较好。
  • 线程安全:使用 ThreadLocal 来存储当前数据源的名称,可以避免多线程之间数据源切换的混乱和不成功的情况。

缺点:

  • 不能同时访问多个数据源:在使用 DynamicDataSource 进行动态数据源切换时,同一时间只能访问一个数据源,不能同时访问多个数据源。
  • 执行效率稍低:在使用 DynamicDataSource 进行动态数据源切换时,每次数据源切换都需要进行一次路由选择,会稍微影响执行效率。
  • 不支持事务嵌套:在使用 DynamicDataSource 进行动态数据源切换时,如果在一个事务中需要访问多个数据源,那么就需要进行事务管理,而 DynamicDataSource 并不支持事务嵌套。

综上所述,使用 DynamicDataSource 和 ThreadLocal 进行动态数据源切换,优点是简单易用、可扩展性强、线程安全,缺点是不能同时访问多个数据源、执行效率稍低、不支持事务嵌套。在具体使用时需要根据业务场景进行选择。

总结

到此这篇关于springboot使用DynamicDataSource动态切换数据源实现的文章就介绍到这了,更多相关springboot动态切换数据源内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot 定制化返回数据的实现示例

    SpringBoot 定制化返回数据的实现示例

    这篇文章主要介绍了SpringBoot 定制化返回数据的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07
  • Java多种方式动态生成doc文档

    Java多种方式动态生成doc文档

    这篇文章主要为大家详细介绍了Java动态生成doc文档的多种方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-09-09
  • mybatis typeAliases 给实体类起别名的方法

    mybatis typeAliases 给实体类起别名的方法

    这篇文章主要介绍了mybatis typeAliases 给实体类起别名,本文给大家分享两种用法,通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • java socket实现局域网聊天

    java socket实现局域网聊天

    这篇文章主要为大家详细介绍了java socket实现局域网聊天,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • Java实现邮件发送遇到的问题

    Java实现邮件发送遇到的问题

    本文给大家分享的是个人在项目过程中,使用Java实现邮件发送的时候所遇到的几个问题以及解决方法,有需要的小伙伴可以参考下
    2016-09-09
  • 教你1秒将本地SpringBoot项目jar包部署到Linux环境(超详细!)

    教你1秒将本地SpringBoot项目jar包部署到Linux环境(超详细!)

    spring Boot简化了Spring应用的开发过程,遵循约定优先配置的原则提供了各类开箱即用(out-of-the-box)的框架配置,下面这篇文章主要给大家介绍了关于1秒将本地SpringBoot项目jar包部署到Linux环境的相关资料,超级详细,需要的朋友可以参考下
    2023-04-04
  • 使用ByteArrayOutputStream写入字符串方式

    使用ByteArrayOutputStream写入字符串方式

    这篇文章主要介绍了使用ByteArrayOutputStream写入字符串方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • Java并发工具之Exchanger线程间交换数据详解

    Java并发工具之Exchanger线程间交换数据详解

    这篇文章主要介绍了Java并发工具之Exchanger线程间交换数据详解,Exchanger是一个用于线程间协作的工具类,Exchanger用于进行线程间的数据交 换,它提供一个同步点,在这个同步点,两个线程可以交换彼此的数据,需要的朋友可以参考下
    2023-12-12
  • 一步步教你JAVA如何优化Elastic Search

    一步步教你JAVA如何优化Elastic Search

    想要榨干Java操作Elasticsearch的所有性能潜力?本指南将一步步教你如何优化Java与Elasticsearch的交互!从此,提升ES查询速度、降低资源消耗不再是难题,赶快一起来探索Java Elasticsearch优化的秘诀吧!
    2024-01-01
  • java中optional的一些常用方法总结

    java中optional的一些常用方法总结

    Java8引入了一个非常强大的特性就是Optional类,其主要解决的问题就是我们编程时常常遇到的空指针异常,下面这篇文章主要给大家介绍了关于java中optional的一些常用方法,需要的朋友可以参考下
    2023-04-04

最新评论