Mybatis使用自定义注解动态配置多数据源

需求说明

目前在项目中有多个数据源,需要多个配置。有好几种办法来配置,比如:

  • jdbcTemplate
  • @DS
  • 自定义注解

下面说明用自定义注解来添加切换多个数据源。

pom的依赖
<dependency>
  <groupId>com.alibaba</groupId>
  <artifactId>druid-spring-boot-starter</artifactId>
  <version>1.1.22</version>
  <exclusions>
    <exclusion>
      <artifactId>spring-boot-autoconfigure</artifactId>
      <groupId>org.springframework.boot</groupId>
    </exclusion>
  </exclusions>
</dependency> 

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-aop</artifactId>
  <version>2.1.3.RELEASE</version>
</dependency>

<dependency>
  <groupId>mysql</groupId>
  <artifactId>mysql-connector-java</artifactId>
  <version>5.1.24</version>
</dependency>

<dependency>
   <groupId>org.aspectj</groupId>
   <artifactId>aspectjweaver</artifactId>
   <version>1.9.4</version>
</dependency>

<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjrt</artifactId>
  <version>1.9.4</version>
</dependency>

<dependency>
  <groupId>org.mybatis.spring.boot</groupId>
  <artifactId>mybatis-spring-boot-starter</artifactId>
  <version>2.1.0</version>
</dependency>

<dependency>
  <groupId>org.mybatis</groupId>
  <artifactId>mybatis</artifactId>
  <version>3.5.2</version>
</dependency>
<dependency>
  <groupId>org.projectlombok</groupId>
  <artifactId>lombok</artifactId>
  <version>1.18.8</version>
</dependency>
配置文件数据库配置信息

多个数据源就配置多个,目前以一个为例。

spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
spring.datasource.size=1

#master
spring.datasource.druid.master.url=jdbc:mysql://localhost:3306/git-demo?useUnicode=true&characterEncoding=utf8&useSSL=false
spring.datasource.druid.master.username=root
spring.datasource.druid.master.password=jiaojiao
spring.datasource.druid.master.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.druid.master.initial-size=5
spring.datasource.druid.master.max-active=30
spring.datasource.druid.master.min-idle=5
spring.datasource.druid.master.max-wait=6000
spring.datasource.druid.master.time-between-eviction-runs-millis=600000
spring.datasource.druid.master.validation-query=SELECT 1
spring.datasource.druid.master.validation-query-timeout=300000
spring.datasource.druid.master.test-on-borrow=true
spring.datasource.druid.master.test-on-return=false
spring.datasource.druid.master.test-while-idle=true
spring.datasource.druid.master.remove-abandoned=true
spring.datasource.druid.master.remove-abandoned-timeout=30
spring.datasource.druid.master.stat-view-servlet.login-username=ytostation
spring.datasource.druid.master.stat-view-servlet.login-password=~!@#$%`12345
mybatis.configuration.map-underscore-to-camel-case=true
mybatis.mapper-locations=classpath*:mapper/*.xml
设置自定义注解

多个数据源就写多个注解。

@Documented
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface MyStyleMasterAnnotation {
}
配置动态数据源:继承AbstractRoutingDataSource

Spring boot提供了AbstractRoutingDataSource 根据用户定义的规则选择当前的数据源,这样我们可以在执行查询之前,设置使用的数据源。实现可动态路由的数据源,在每次数据库查询操作前执行。它的抽象方法 determineCurrentLookupKey() 决定使用哪个数据源。

查看源码可知:AbstractRoutingDataSource的多数据源动态切换的核心逻辑是:在程序运行时,把数据源数据源通过 AbstractRoutingDataSource 动态织入到程序中,灵活的进行数据源切换。
基于AbstractRoutingDataSource的多数据源动态切换,可以实现读写分离,这么做缺点也很明显,无法动态的增加数据源。

public class MyStyleRoutingDataSource extends AbstractRoutingDataSource {
    @Override
    protected Object determineCurrentLookupKey() {
        return getDbType();
    }

    public enum DbType{
        MASTER
    }

    private static final ThreadLocal<DbType> contextHolder = new ThreadLocal<>();
    public static void setDbType(DbType dbType){
        if (dbType == null){
            throw new NullPointerException();
        }
        contextHolder.set(dbType);
    }

    public static DbType getDbType(){
        return contextHolder.get() ==null ? DbType.MASTER :  contextHolder.get();
    }

    public static void clearDbType(){
        contextHolder.remove();
    }
}
绑定自定义注解

每个注解一个切面。

@Component
@Slf4j
public class MyStyleMasterAspect {
    @Around("@annotation(myStyleMasterAnnotation)")
    public Object invoke(ProceedingJoinPoint joinPoint, MyStyleMasterAnnotation myStyleMasterAnnotation) throws Throwable {
        try {           	
 MyStyleRoutingDataSource.setDbType(MyStyleRoutingDataSource.DbType.MASTER);
            return joinPoint.proceed();
        } finally {
            MyStyleRoutingDataSource.clearDbType();
        }
    }
}
数据源配置类
@Configuration
@Slf4j
@EnableTransactionManagement
public class MyStyleDataSourceConfiguration extends MybatisAutoConfiguration {

    @Value("${spring.datasource.type}")
    private Class<? extends DataSource> dbSourceType;

    public MyStyleDataSourceConfiguration(MybatisProperties properties, ObjectProvider<Interceptor[]> interceptorsProvider, ObjectProvider<TypeHandler[]> typeHandlersProvider, ObjectProvider<LanguageDriver[]> languageDriversProvider, ResourceLoader resourceLoader, ObjectProvider<DatabaseIdProvider> databaseIdProvider, ObjectProvider<List<ConfigurationCustomizer>> configurationCustomizersProvider) {
        super(properties, interceptorsProvider, typeHandlersProvider, languageDriversProvider, resourceLoader, databaseIdProvider, configurationCustomizersProvider);
    }
		
  	//多个数据源就再多写多个实例
    @Bean
    @Primary
    @ConfigurationProperties(prefix = "spring.datasource.druid.master")
    public DataSource masterDataSource(){
        return DataSourceBuilder.create().type(dbSourceType).build();
    }

    @Bean
    @Override
    public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
        return super.sqlSessionFactory(dataSource());
    }

    @Bean
    public DataSource dataSource(){
        AbstractRoutingDataSource routingDataSource = new MyStyleRoutingDataSource();

        Map<Object,Object> targetDataSources = new HashMap<>(5);
				//多少个数据源就添加到map中
       targetDataSources.put(MyStyleRoutingDataSource.DbType.MASTER,masterDataSource());
        routingDataSource.setTargetDataSources(targetDataSources);
        routingDataSource.afterPropertiesSet();
        return routingDataSource;
    }
}
启动类中添加@MapperScan
@MapperScan(basePackages = {"cn.mystyle.*.dao"})
涉及到的mapper的xml文件
<select id="queryStudentInfo" resultType="cn.mystyle.springboot.entity.Student">
  select id,name,id_card_no,address from student
</select>
实体类
@Data
@NoArgsConstructor
@AllArgsConstructor
@ToString
public class Student {
    private int id;
    private String name;
    private String idCardNo;
    private String address;
}
测试类
/**
*测试controller类
*/
@RestController
@Slf4j
public class TestController {
	
    @Resource
    private IMyStyleMasterService myStyleMasterService;
    
	@GetMapping("/test/queryStudentInfo")
    public String queryStudentInfo(){
      List<Student> studentList = myStyleMasterService.queryStudentInfo();
      log.info("size:{}",studentList.size());
      return "success";
    }
}

/**
*涉及db的接口
*/
public interface IMyStyleMasterService {
    List<Student> queryStudentInfo();
}

/**
*直接调用
*/
@Component
public class MyStyleMasterServiceImpl implements IMyStyleMasterService {
    @Resource
    private StudentDao studentDao;

    @MyStyleMasterAnnotation
    @Override
    public List<Student> queryStudentInfo() {
        return studentDao.queryStudentInfo();
    }
}
测试结果

经过测试可以看出来测试成功。
在这里插入图片描述

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值