SpringBoot中多数据源配置

本文介绍了如何在Spring Boot中配置和使用动态数据源,包括数据源信息配置、数据源切换注解、枚举、数据库持有、数据源路由、动态数据源配置、JDBC配置和AOP切面。在实际应用中,数据源切换在同一个事务中出现失败的问题,这可能涉及到事务的一致性和数据源管理的复杂性。
摘要由CSDN通过智能技术生成

1、数据源信息配置

spring:
  datasource:
    master:
      url:  jdbc:mysql://127.0.0.1:3306/master?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&failOverReadOnly=false&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai
      username: demo2_admin
      password: xxxxxxxx
      driver-class-name: com.mysql.cj.jdbc.Driver
    slaver:
      url:  jdbc:mysql://127.0.0.1:3306/slaver?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&failOverReadOnly=false&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai
      username: demo1_admin
      password: yyyyyyyy
      driver-class-name: com.mysql.cj.jdbc.Driver

2、数据源切换注解

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface DB{

    EmDSType value() default EmDSType.MASTER;

}

3、数据源枚举

public enum EmDSType {

    //生产环境
    MASTER("master"),
    //预发环境
    SLAVER("slaver");

    private String name;

    EmDSType(String name) {
        this.name = name;
    }


    public void setName(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }
}

4、数据库持有

public class DataSourceHolder {

    private static final ThreadLocal<String> DS_HOLDER = new ThreadLocal<>();

    /**
    * 设置数据源
    * @param dsType
    */
    public static void setDataSource(EmDSType dsType) {
        DS_HOLDER.set(dsType.getName());
    }

    /**
    * 获取数据源
    * @return
    */
    public static String getDataSource() {
        return DS_HOLDER.get();
    }

    /**
    * 清理数据源
    */
    public static void clearDataSource() {
        DS_HOLDER.remove();
    }
}

5、数据源路由:AbstractRoutingDataSource

public class DynamicDataSource extends AbstractRoutingDataSource {

    private static final Logger LOGGER = LoggerFactory.getLogger(DynamicDataSource.class);

    @Override
    protected Object determineCurrentLookupKey() {
        LOGGER.info("切换数据源:{}", DataSourceHolder.getDataSource());
        return DataSourceHolder.getDataSource();
    }
}

6、动态数据源配置

@Configuration
public class DataSourceConfig {


    /**
    * 主数据源
    * @return
    */master")
    @ConfigurationProperties("spring.datasource.master")
    public DataSource master() {
        return new DruidDataSource();
    }

    /**
    * 从数据源
    * @return
    */
    @Bean("slaver")
    @ConfigurationProperties("spring.datasource.slaver")
    public DataSource slaver() {
        return new DruidDataSource();
    }

    /**
    * 优先加载数据源
    * @return
    */
    @Bean("dynamicDataSource")
    @Primary
    public DataSource dynamicDataSource() {
        DynamicDataSource dataSource = new DynamicDataSource();
        dataSource.setDefaultTargetDataSource(master());
        Map<Object, Object> dataSourceMap = Maps.newHashMap();
        dataSourceMap.put(EmDSType.MASTER.getName(), master());
        dataSourceMap.put(EmDSType.SLAVER.getName(), slaver());
        dataSource.setTargetDataSources(dataSourceMap);
        return dataSource;
    }

    /**
    * 事务管理
    * @return
    */
    public PlatformTransactionManager transactionManager() {
        return new DataSourceTransactionManager(dynamicDataSource());
    }

}

7、JDBC配置

@Configuration
public class JdbcTemplateConfig {

@Bean
@Primary
JdbcTemplate jdbcTemplateInner(@Qualifier("inner")DataSource inner) {
return new JdbcTemplate(inner);
}

@Bean
JdbcTemplate jdbcTemplateOuter(@Qualifier("outer")DataSource outer) {
return new JdbcTemplate(outer);
}
}

8、AOP切面

@Aspect
@Component
@Order(1)
public class DataSourceAspect {

    private static final Logger LOGGER = LoggerFactory.getLogger(DataSourceAspect.class);

    @Pointcut("@annotation(com.jd.jx.lowcode.datasource.annotation.DB)")
    public void pointCut() {}

    @Before("pointCut()")
    public void before(JoinPoint joinPoint) {
        Object target = joinPoint.getTarget();
        MethodSignature signature = (MethodSignature)joinPoint.getSignature();
        EmDSType dataSource = EmDSType.PROD;
        try {
            Class<?> aClass = target.getClass();
            if (aClass.isAnnotationPresent(DB.class)) {
                DB annotation = aClass.getAnnotation(DB.class);
                dataSource = annotation.value();
            }else {
                Method method = signature.getMethod();
                if (method.isAnnotationPresent(DB.class)) {
                    DB annotation = method.getAnnotation(DB.class);
                    dataSource = annotation.value();
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        LOGGER.info("aop切换数据源:{}", dataSource.getName());

        DataSourceHolder.setDataSource(dataSource);
    }

    @After("pointCut()")
    public void after() {
        DataSourceHolder.clearDataSource();
    }
}

9、问题

在同一个事物中,数据源切换失败

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot配置多个Redis数据源需要做以下几个步骤: 1. 在pom.xml添加Redis和Lettuce的依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> <dependency> <groupId>io.lettuce</groupId> <artifactId>lettuce-core</artifactId> </dependency> ``` 2. 在application.yml配置多个Redis数据源,例如: ```yaml spring: redis: host: localhost port: 6379 password: password database: 0 jedis: pool: max-active: 8 max-idle: 8 min-idle: 0 max-wait: -1ms redis2: host: localhost port: 6380 password: password database: 1 lettuce: pool: max-active: 8 max-idle: 8 min-idle: 0 max-wait: -1ms ``` 注意,每个Redis数据源需要定义一个唯一的名称,如redis和redis2。 3. 在Java代码创建多个RedisTemplate实例,并指定对应的RedisConnectionFactory,例如: ```java @Configuration public class RedisConfig { @Bean(name = "redisTemplate") public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) { RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(redisConnectionFactory); redisTemplate.setKeySerializer(new StringRedisSerializer()); redisTemplate.setValueSerializer(new Jackson2JsonRedisSerializer<>(Object.class)); return redisTemplate; } @Bean(name = "redisTemplate2") public RedisTemplate<String, Object> redisTemplate2(@Qualifier("redisConnectionFactory2") RedisConnectionFactory redisConnectionFactory) { RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(redisConnectionFactory); redisTemplate.setKeySerializer(new StringRedisSerializer()); redisTemplate.setValueSerializer(new Jackson2JsonRedisSerializer<>(Object.class)); return redisTemplate; } @Bean(name = "redisConnectionFactory") public RedisConnectionFactory redisConnectionFactory(RedisProperties redisProperties) { JedisConnectionFactory jedisConnectionFactory = new JedisConnectionFactory(); jedisConnectionFactory.setHostName(redisProperties.getHost()); jedisConnectionFactory.setPort(redisProperties.getPort()); jedisConnectionFactory.setPassword(redisProperties.getPassword()); jedisConnectionFactory.setDatabase(redisProperties.getDatabase()); jedisConnectionFactory.setUsePool(true); jedisConnectionFactory.getPoolConfig().setMaxTotal(redisProperties.getJedis().getPool().getMaxActive()); jedisConnectionFactory.getPoolConfig().setMaxIdle(redisProperties.getJedis().getPool().getMaxIdle()); jedisConnectionFactory.getPoolConfig().setMinIdle(redisProperties.getJedis().getPool().getMinIdle()); jedisConnectionFactory.getPoolConfig().setMaxWaitMillis(redisProperties.getJedis().getPool().getMaxWait().toMillis()); return jedisConnectionFactory; } @Bean(name = "redisConnectionFactory2") public RedisConnectionFactory redisConnectionFactory2(@Qualifier("redisProperties2") RedisProperties redisProperties) { LettuceConnectionFactory lettuceConnectionFactory = new LettuceConnectionFactory(); lettuceConnectionFactory.setHostName(redisProperties.getHost()); lettuceConnectionFactory.setPort(redisProperties.getPort()); lettuceConnectionFactory.setPassword(redisProperties.getPassword()); lettuceConnectionFactory.setDatabase(redisProperties.getDatabase()); lettuceConnectionFactory.setUsePooling(true); lettuceConnectionFactory.setPoolConfig(getLettucePoolConfig(redisProperties.getLettuce().getPool())); return lettuceConnectionFactory; } @Bean(name = "redisProperties") @ConfigurationProperties(prefix = "spring.redis") public RedisProperties redisProperties() { return new RedisProperties(); } @Bean(name = "redisProperties2") @ConfigurationProperties(prefix = "spring.redis2") public RedisProperties redisProperties2() { return new RedisProperties(); } private GenericObjectPoolConfig<?> getLettucePoolConfig(LettucePoolingClientConfiguration.Pool pool) { GenericObjectPoolConfig<?> config = new GenericObjectPoolConfig<>(); config.setMaxTotal(pool.getMaxActive()); config.setMaxIdle(pool.getMaxIdle()); config.setMinIdle(pool.getMinIdle()); config.setMaxWaitMillis(pool.getMaxWait().toMillis()); return config; } } ``` 要使用第二个Redis数据源,只需使用@Qualifier指定对应的RedisTemplate实例,例如: ```java @Autowired @Qualifier("redisTemplate2") private RedisTemplate<String, Object> redisTemplate2; ``` 这样就可以在Spring Boot配置多个Redis数据源了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值