SpringBoot 如何实现多数据源

本文详细介绍了在SpringBoot中配置和管理多个数据源的步骤,包括YML配置、数据源bean创建、动态数据源实现以及AOP切面进行数据源切换。示例代码展示了如何使用自定义注解在不同操作中选择合适的数据源,适用于需要同时操作多个数据库的场景。
摘要由CSDN通过智能技术生成

SpringBoot 如何实现多数据源

第一步:配置yml

spring:
  datasource:
    type: com.alibaba.druid.pool.DruidDataSource
    datasource1:
      url: jdbc:mysql://127.0.0.1:3306/datasource1?serverTimezone=UTC&useUnicode=true&characterEncoding=UTF8&useSSL=false
      username: root
      password: 123456
      driver-class-name: com.mysql.cj.jdbc.Driver
    datasource2:
      url: jdbc:mysql://127.0.0.1:3306/datasource2?serverTimezone=UTC&useUnicode=true&characterEncoding=UTF8&useSSL=false
      username: root
      password: 123456
      driver-class-name: com.mysql.cj.jdbc.Driver

第二步:创建一个配置类

@Configuration
public class DataSourceConfig {

    @Bean
    @ConfigurationProperties(prefix = "spring.datasource.datasource1")
    public DataSource dataSource1() {
        // 底层会自动拿到spring.datasource中的配置, 创建一个DruidDataSource
        return DruidDataSourceBuilder.create().build();
    }

    @Bean
    @ConfigurationProperties(prefix = "spring.datasource.datasource2")
    public DataSource dataSource2() {
        // 底层会自动拿到spring.datasource中的配置, 创建一个DruidDataSource
        return DruidDataSourceBuilder.create().build();
    }


	//事务管理器
    @Bean
    public DataSourceTransactionManager transactionManager1(DynamicDataSource dataSource){
        DataSourceTransactionManager dataSourceTransactionManager = new DataSourceTransactionManager();
        dataSourceTransactionManager.setDataSource(dataSource);
        return dataSourceTransactionManager;
    }

    @Bean
    public DataSourceTransactionManager transactionManager2(DynamicDataSource dataSource){
        DataSourceTransactionManager dataSourceTransactionManager = new DataSourceTransactionManager();
        dataSourceTransactionManager.setDataSource(dataSource);
        return dataSourceTransactionManager;
    }
}

第三步:继承AbstractRoutingDataSource 重新determineTargetDataSource方法

@Component
@Primary   // 将该Bean设置为主要注入Bean
public class DynamicDataSource extends AbstractRoutingDataSource {


    // 当前使用的数据源标识
    public static ThreadLocal<String> name=new ThreadLocal<>();

    // 注入数据源
    @Autowired
    DataSource dataSource1;
    @Autowired
    DataSource dataSource2;


    // 返回当前数据源标识
    @Override
    protected Object determineCurrentLookupKey() {
        return name.get();

    }

    /*
    * 此方法在是Spring容器启动时执行,目的是将自定义的数据源放入targetDataSources 这个map中
    * 通过父类的setTargetDataSources方法设置成员变量,
    * 注意:必须调用父类的afterPropertiesSet()方法,这个方法会将targetDataSources赋给resolvedDataSources
    * 切换数据时会从resolvedDataSources中查找
    *
    * */
    @Override
    public void afterPropertiesSet() {

        // 为targetDataSources初始化所有数据源
        Map<Object, Object> targetDataSources=new HashMap<>();
        targetDataSources.put("W",dataSource1);
        targetDataSources.put("R",dataSource2);

        // 赋给成员变量targetDataSources
        super.setTargetDataSources(targetDataSources);

        // 为defaultTargetDataSource 设置默认的数据源
        super.setDefaultTargetDataSource(dataSource1);

        //调用父类的afterPropertiesSet()方法
        super.afterPropertiesSet();
    }
}

第四步: 多数据源切换方式,AOP+自定义注解

// 自定义注解
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface WR {
	// 默认值为W
    String value() default "W";
}

//AOP
@Component
@Aspect
public class DynamicDataSourceAspect implements Ordered {

    // 前置
    @Before("within(com.xxxxx.service.impl.*) && @annotation(wr)")
    public void before(JoinPoint point, WR wr){
    	// 获取注解的值
        String name = wr.value();
        // 设置使用那个数据源
        DynamicDataSource.name.set(name);
    }
}

第五步:测试

@Service
public class UserService implements UserService {

    @Autowired
    UserMapper userMapper;


    @Override
    @WR("R")        // 库2
    public List<User> list() {
        return userMapper.list();
    }

    @Override
    @WR("W")        // 库1
    public void save(User user) {
        userMapper.save(user);
    }

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值