springboot动态使用DruidDataSource切换数据源(动态配置多个数据源)

1、添加依赖,在pom文件中添加

 <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
            <version>2.5.0</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.2.16</version>
        </dependency>

2、添加数据源名字定义,这个可有可无,没有的话,需要在切面和运用中定义

public class DataSourceConstants {
    /**
     * master数据源
     */
    public static final String DS_KEY_MASTER = "master";
    /**
     * slave数据源
     */
    public static final String DS_KEY_SLAVE = "slave";
}

3、添加注解类

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

// 标记注解可使用在方法与类上
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface DS {
    // 默认值为MASTER
    String value() default DataSourceConstants.DS_KEY_MASTER;
}

4、动态数据源的切换定义

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

/**
 * 动态数据源
 **/
public class DynamicDataSource extends AbstractRoutingDataSource {

    @Override
    protected Object determineCurrentLookupKey() {
        return DynamicDataSourceContextHolder.getContextKey();
    }
}

5、动态切面实现类

import cn.com.yixiukeji.app.ServiceContext;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.stereotype.Component;

/**
 * 切面
 */
@Aspect
@Component
//@Order(-10)
@ConditionalOnProperty(
        prefix = "yixiu.cube.center.multi-datasource",
        value = {"enable"},
        havingValue = "true"
)
public class DynamicDataSourceAspect {
    // 以在类上使用了@Service作为切入点
    @Pointcut("@within(org.springframework.stereotype.Service)")
    public void dataSourcePointCut() {
    }

    @Around("dataSourcePointCut()")
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Class<?> aClass = Class.forName(signature.getDeclaringType().getName());
        // 方法优先,如果方法上存在注解,则优先使用方法上的注解
        if (signature.getMethod().isAnnotationPresent(DS.class)) { DynamicDataSourceContextHolder.setContextKey(signature.getMethod().getAnnotation(DS.class).value());
            // 其次类优先,如果类上存在注解,则使用类上的注解
        }else  if (aClass.isAnnotationPresent(DS.class)) { DynamicDataSourceContextHolder.setContextKey(aClass.getAnnotation(DS.class).value());
            // 如果都不存在,则使用默认
        }   else {
        //默认选择源
            DynamicDataSourceContextHolder.setContextKey("slave");
        }
        System.out.println(ServiceContext.getContext().get("yes.req.instanceId"));
        try {
            return joinPoint.proceed();
        } finally {
            DynamicDataSourceContextHolder.removeContextKey();
        }
    }
}

6、动态数据源配置

/**
 * 动态数据源配置
 **/
@EnableAutoConfiguration(exclude = { DataSourceAutoConfiguration.class })
@Configuration
@ConditionalOnProperty(
        prefix = "yixiu.cube.center.multi-datasource",
        value = {"enable"},
        havingValue = "true"
)
@MapperScan(basePackages = "cn.com.dao.mapper")
public class DynamicDataSourceConfig {
    @Bean(DataSourceConstants.DS_KEY_MASTER)
    // 需要与配置文件中对应
    @ConfigurationProperties(prefix = "yixiu.cube.center.multi-datasource.datasource.master")
    public DruidDataSource masterDataSource() {
        return DruidDataSourceBuilder.create().build();
    }

    @Bean(DataSourceConstants.DS_KEY_SLAVE)
    @ConfigurationProperties(prefix = "yixiu.cube.center.multi-datasource.datasource.slave")
    public DruidDataSource slaveDataSource() {
        return DruidDataSourceBuilder.create().build();
    }

    @Bean
    @ConfigurationProperties(
            prefix = "yixiu.cube.center.multi-datasource"
    )
    @ConditionalOnMissingBean
    public MultiDataSourceProperties multiDatasourceProperties() {
        return new MultiDataSourceProperties();
    }

    @Bean
    @ConditionalOnMissingBean
    public DataSource dataSource(MultiDataSourceProperties multiDatasourceProperties) {
        DynamicRoutingDataSource dataSource = new DynamicRoutingDataSource();
        multiDatasourceProperties.getDatasource().forEach((name, dataSourceVo) -> {
            DataSource data = DataSourceBuilder.create().url(dataSourceVo.getUrl())
                    .username(dataSourceVo.getUsername()).password(dataSourceVo.getPassword())
                    .driverClassName(dataSourceVo.getDriverClassName()).build();
            dataSource.addDataSource(name,data);
        });
        return dataSource;
    }


    @Bean
    @Primary
    public DynamicDataSource dynamicDataSource(MultiDataSourceProperties multiDataSourceProperties) {
        Map<Object, Object> dataSourceMap = new ConcurrentHashMap<>(multiDataSourceProperties.getDatasource().size());
        DynamicRoutingDataSource dataSource = new DynamicRoutingDataSource();
        multiDataSourceProperties.getDatasource().forEach((name, dataSourceVo) -> {
            DataSource data = DataSourceBuilder.create().url(dataSourceVo.getUrl())
                    .username(dataSourceVo.getUsername()).password(dataSourceVo.getPassword())
                    .driverClassName(dataSourceVo.getDriverClassName()).build();
            dataSource.addDataSource(name,data);
            dataSourceMap.put(name,data);
        });
//这种配置是定死的,不好玩
//        dataSourceMap.put(DataSourceConstants.DS_KEY_MASTER, masterDataSource());
//        dataSourceMap.put(DataSourceConstants.DS_KEY_SLAVE, slaveDataSource());
        //设置动态数据源
        DynamicDataSource dynamicDataSource = new DynamicDataSource();
        dynamicDataSource.setTargetDataSources(dataSourceMap);
        //dynamicDataSource.setDefaultTargetDataSource(masterDataSource());
        //dynamicDataSource.setDefaultTargetDataSource(multiDataSourceProperties.getDatasource().get("master"));
        return dynamicDataSource;
    }
}

7、动态数据源名称上下文处理

/**
 * 动态数据源名称上下文处理
 **/
public class DynamicDataSourceContextHolder {

    /**
     * 动态数据源名称上下文
     */
    private static final ThreadLocal<String> DATASOURCE_CONTEXT_KEY_HOLDER = new ThreadLocal<>();

    /**
     * 设置数据源
     * @param key
     */
    public static void setContextKey(String key){
        DATASOURCE_CONTEXT_KEY_HOLDER.set(key);
    }

    /**
     * 获取数据源名称
     * @return
     */
    public static String getContextKey(){
        String key = DATASOURCE_CONTEXT_KEY_HOLDER.get();
        return key == null?DataSourceConstants.DS_KEY_MASTER:key;
    }

    /**
     * 删除当前数据源名称
     */
    public static void removeContextKey(){
        DATASOURCE_CONTEXT_KEY_HOLDER.remove();
    }
}

8、动态配置的映射,可以自定义

import java.util.Map;
public class MultiDataSourceProperties {
    private String primary = "master";
    private Map<String, MultiDataSourceVo> datasource;

    public  MultiDataSourceProperties() {
    }

    public String getPrimary() {
        return this.primary;
    }

    public void setPrimary(String primary) {
        this.primary = primary;
    }

    public Map<String, MultiDataSourceVo> getDatasource() {
        return this.datasource;
    }

    public void setDatasource(Map<String, MultiDataSourceVo> datasource) {
        this.datasource = datasource;
    }

    public MultiDataSourceVo getPrimaryMultiDataSourceVo() {
        return this.datasource != null && !this.datasource.isEmpty() ? (MultiDataSourceVo)this.datasource.get(this.primary) : null;
    }
}

9、动态属性的实体类

import lombok.Data;

@Data
public class MultiDataSourceVo {
    private String driverClassName = "com.mysql.cj.jdbc.Driver";
    private String url;
    private String username;
    private String password;
}

10、yaml文件的配置

yixiu.cube.center.multi-datasource:
        enable: true #开启多数据源配置,关闭单个数据源配置
        primary: master #设置默认的数据源或者数据源组,默认值即为master
        datasource:
          master:
            driver-class-name: com.mysql.cj.jdbc.Driver # 3.2.0开始⽀持SPI可省略此配
            url: jdbc:mysql://aliyuncs.com:3306/test?useUnicode=true&characterEncoding=utf-8&useSSL=false&allowMultiQueries=true&serverTimezone=GMT%2b8
            username: xkc
            password: 123456
          slave:
            url: jdbc:mysql://192.168.2.2:30306/test?useUnicode=true&characterEncoding=utf-8&useSSL=false&allowMultiQueries=true&serverTimezone=GMT%2b8
            username: xkc
            password: 123456
            driver-class-name: com.mysql.cj.jdbc.Driver
          slave1:
            url: jdbc:mysql://192.168.2.2:30306/test1?useUnicode=true&characterEncoding=utf-8&useSSL=false&allowMultiQueries=true&serverTimezone=GMT%2b8
            username: xkc
            password: 123456
            driver-class-name: com.mysql.cj.jdbc.Driver

11、编写接口添加数据就可以在目标数据库查看了,这个简单就不说明了。其他的配置,可以查看官方说明或者博客资料。

12、参考项目:https://gitee.com/baomidou/dynamic-datasource-spring-boot-starter

好的,以下是一个使用Spring Boot和Druid实现多数据源的示例: 首先,在pom.xml文件中添加依赖: ```xml <dependency> <groupId>com.alibaba</groupId> <artifactId>druid-spring-boot-starter</artifactId> <version>1.2.6</version> </dependency> ``` 然后,在application.properties或application.yml文件中配置Druid数据源: ```yaml spring: datasource: master: driver-class-name: com.mysql.jdbc.Driver url: jdbc:mysql://localhost:3306/master_db?useUnicode=true&characterEncoding=utf-8 username: root password: root type: com.alibaba.druid.pool.DruidDataSource initialSize: 5 minIdle: 5 maxActive: 20 maxWait: 60000 slave: driver-class-name: com.mysql.jdbc.Driver url: jdbc:mysql://localhost:3306/slave_db?useUnicode=true&characterEncoding=utf-8 username: root password: root type: com.alibaba.druid.pool.DruidDataSource initialSize: 5 minIdle: 5 maxActive: 20 maxWait: 60000 ``` 接下来,创建两个数据源配置类: ```java @Configuration @MapperScan(basePackages = "com.example.demo.mapper.master", sqlSessionTemplateRef = "masterSqlSessionTemplate") public class MasterDataSourceConfig { @Bean(name = "masterDataSource") @ConfigurationProperties(prefix = "spring.datasource.master") public DataSource dataSource() { return DruidDataSourceBuilder.create().build(); } @Bean(name = "masterSqlSessionFactory") public SqlSessionFactory sqlSessionFactory(@Qualifier("masterDataSource") DataSource dataSource) throws Exception { SqlSessionFactoryBean bean = new SqlSessionFactoryBean(); bean.setDataSource(dataSource); return bean.getObject(); } @Bean(name = "masterTransactionManager") public DataSourceTransactionManager transactionManager(@Qualifier("masterDataSource") DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } @Bean(name = "masterSqlSessionTemplate") public SqlSessionTemplate sqlSessionTemplate(@Qualifier("masterSqlSessionFactory") SqlSessionFactory sqlSessionFactory) { return new SqlSessionTemplate(sqlSessionFactory); } } ``` ```java @Configuration @MapperScan(basePackages = "com.example.demo.mapper.slave", sqlSessionTemplateRef = "slaveSqlSessionTemplate") public class SlaveDataSourceConfig { @Bean(name = "slaveDataSource") @ConfigurationProperties(prefix = "spring.datasource.slave") public DataSource dataSource() { return DruidDataSourceBuilder.create().build(); } @Bean(name = "slaveSqlSessionFactory") public SqlSessionFactory sqlSessionFactory(@Qualifier("slaveDataSource") DataSource dataSource) throws Exception { SqlSessionFactoryBean bean = new SqlSessionFactoryBean(); bean.setDataSource(dataSource); return bean.getObject(); } @Bean(name = "slaveTransactionManager") public DataSourceTransactionManager transactionManager(@Qualifier("slaveDataSource") DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } @Bean(name = "slaveSqlSessionTemplate") public SqlSessionTemplate sqlSessionTemplate(@Qualifier("slaveSqlSessionFactory") SqlSessionFactory sqlSessionFactory) { return new SqlSessionTemplate(sqlSessionFactory); } } ``` 最后,在需要使用的Mapper类中指定使用哪个数据源: ```java @Repository public interface UserMapper { // 使用master数据源 @DataSource("masterDataSource") List<User> selectAllFromMaster(); // 使用slave数据源 @DataSource("slaveDataSource") List<User> selectAllFromSlave(); } ``` 这样就可以通过@DataSource注解来动态切换数据源了。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值