一、引入相关maven配置

mybatis;  mysql驱动;jdbc

         <dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.1.1</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.18</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
<version>2.2.2.RELEASE</version>
</dependency>

二、编写配置文件

配置两个数据源:我这里配置了一个腾讯云的数据库和一个本地的数据库

分别指定四个字段:数据库链接、用户名、密码、驱动

 三、编写数据库类型枚举

public enum DataSourceEnum {

    TENCENT(0),LOCAL(1);

    private  int value;

    private DataSourceEnum(int _value)
{
this.value=_value;
}
}

四、编写配置文件读取帮助类

注:我这里做这一步是为了更加透明的展示配置数据源的过程,如果想偷懒,这一一步可以不做,直接通过注解读取配置文件注入到容器中

DataSourceElement 数据库链接实体
/*数据库链接实体类
* 字段和配置文件保持一致*/
public class DataSourceElement { private String url; private String userName; private String password; private String classDirver;
//忽略getter setter
DataSourceItem 配置读取帮助类
@Component
@ConfigurationProperties(prefix = "spring.datasource")
public class DataSourceItem {
//腾讯云链接 对应配置文件 tencent-db
private DataSourceElement tencentDb;
//本地链接 对应配置文件 local-db
private DataSourceElement localDb;
//忽略getter serrer
}

五、编写获取当前线程中执行的数据源的帮助类

这个类是为了指定我们当前线程中需要使用的数据源

public class DynamicDataSourceManger {

    /*
* 初始化当前线程数据库类型*/
private static final ThreadLocal<DataSourceEnum> threadLocalDbType = new ThreadLocal<DataSourceEnum>() {
@Override
protected DataSourceEnum initialValue() {
return DataSourceEnum.TENCENT;
} ;
}; /*获取当前线程数据库*/
public static DataSourceEnum get() {
return threadLocalDbType.get();
} /*设置当前线程数据库*/
public static void set(DataSourceEnum dataSourceEnum) {
threadLocalDbType.set(dataSourceEnum);
} /*清楚当前线程数据库缓存*/
public static void reset() {
threadLocalDbType.remove();
}
}

六、继承AbstractRoutingDataSource

AbstractRoutingDataSource是spring提供的动态切换数据源工具类,他可以实现运行期间的数据源路由切换,具体内容请查看官方文档,我这里只写怎么用
简单的来说他会通过我们指定的key来获取我们设置的不同数据源,数据源的配置会在下面写到。

这里通过我们上面编写的DynamicDataSourceManger 拿到了当前线程需要执行的数据源的key
public class DynamicDataSource extends AbstractRoutingDataSource {
@Override
protected Object determineCurrentLookupKey() {
/*读取当前线程的数据源key值配置*/
return DynamicDataSourceManger.get();
}
}

七、配置多数据源以及AbstractRoutingDataSource的实现

注:下面的代码中我使用了通过我们在第四部编写的配置文件帮助类来获取配置文件的内容实现了Bean的注入,这么做是为了方便读者明白DataSource的创建过程

也可以直接使用 @ConfigurationProperties(prefix = "spring.datasource.tencent-db")直接装配上去,这样的话只需要写一行

DataSource dataSource = DataSourceBuilder.create().build();即可

注意事项:1.我们需要为不通的数据源指定不通的Bean名称,防止冲突
2.需要在获取AbstractRoutingDataSource实现的Bean上添加@Primary
3.编写完后需要关闭spring的自动配置数据源,否则会引发循环依赖(下面会写)

Spring容器注入类
/*数据源注入*/
@Configuration
public class DynamicDataSourceConfiguration { /*拿到数据库链接*/
@Autowired
DataSourceItem dataSourceItem; /*注入腾讯云数据库*/
@Bean(name = "TENCENTDB")
public DataSource getTencentDataSource() {
DataSourceElement tencentSoure = dataSourceItem.getTencentDb();
DataSource dataSource = DataSourceBuilder.create()
.url(tencentSoure.getUrl())
.username(tencentSoure.getUserName())
.password(tencentSoure.getPassword())
.driverClassName(tencentSoure.getClassDirver())
.build();
return dataSource;
} /*注入本地数据库*/
@Bean(name = "LOCALDB")
public DataSource getLocalDataSource() {
DataSourceElement localSource = dataSourceItem.getLocalDb();
DataSource dataSource = DataSourceBuilder.create()
.url(localSource.getUrl())
.username(localSource.getUserName())
.password(localSource.getPassword())
.driverClassName(localSource.getClassDirver())
.build();
return dataSource;
} /*注入AbstractRoutingDataSource*/
@Bean
@Primary
public DynamicDataSource getDyanmiceDataSource(@Qualifier("TENCENTDB") DataSource tencentDb,
@Qualifier("LOCALDB") DataSource localDb) { Map<Object,Object> hashMap=new HashMap<Object, Object>();
hashMap.put(DataSourceEnum.TENCENT,tencentDb);
hashMap.put(DataSourceEnum.LOCAL,localDb); DynamicDataSource dataSource=new DynamicDataSource();
/*设置数据源*/
dataSource.setTargetDataSources(hashMap);
/*设置默认数据源*/
dataSource.setDefaultTargetDataSource(tencentDb);
/*调用方法实现数据源配置*/
dataSource.afterPropertiesSet();
return dataSource;
} }
更改Spring Boot启动设置,关闭数据源自动注入和手动把我们的配置类注入到IOC容器中
@Import(DynamicDataSourceConfiguration.class)
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})
public class DynamicApplication { public static void main(String[] args) {
SpringApplication.run(DynamicApplication.class, args);
} }

八、编写实体类、Mybatis的Mapper和XML、配置Mybatis

这里我在腾讯云建了一个叫t_user的表,在本地库建了一个叫person的表

我们分别编写他们的Mapper和实体以及XML。

以下代码皆免去get和set,并且把代码都写在了一起方便对比,实际的时候是拆开写的(这是基础吧。。。。应该不用我说)

public class Person {

    private Integer id;

    private  String name;

    private String address;

    private String province;

    private  String city;

    private  String area;
}
public class User { private Integer id; private String name; private String sex; private String address; private String city; private String area;
}

Mapper

@Mapper
public interface PersonMapper { List<Person> selectAllPerson();
} @Mapper
public interface UserMapper { List<User> selectAllUser();
}

XML

<mapper namespace="com.example.dynamic.mapper.PersonMapper">
<select id="selectAllPerson" resultType="com.example.dynamic.model.Person">
select * from person
</select>
</mapper> <mapper namespace="com.example.dynamic.mapper.UserMapper">
<select id="selectAllUser" resultType="com.example.dynamic.model.User">
select * from t_user
</select>
</mapper>

接口

我们在两个接口中,分别调用了腾讯云和本地两个数据源,在调用前通过在第五部编写的DynamicDataSourceManger来实现当前线程数据源key的切换

@RestController
@RequestMapping("/hello")
public class DynamicDataSourceController { @Autowired
UserMapper userMapper; @Autowired
PersonMapper personMapper; @RequestMapping("/user")
public List<User> selectAllUser(){
DynamicDataSourceManger.set(DataSourceEnum.TENCENT);
return userMapper.selectAllUser();
} @RequestMapping("/person")
public List<Person> selectAllPerson(){
DynamicDataSourceManger.set(DataSourceEnum.LOCAL);
return personMapper.selectAllPerson();
}
}

九、调用接口

调用腾讯云数据源

调用本地数据源

我们发现,这时候已经实现了多数据源的切换,这种写法其实有些生硬,需要手动设置数据源,可以通过AOP的方式把数据源设置编进去

十:编写Aspect实现动态切换

引入pom

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

自定义注解TargetDataSource,标明只能在函数上标注

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TargetDataSource { String value() default "";
}

自定义AOP切面类

@Component
@Aspect
public class DynamicSourceAspect { @Around("execution(* com.example.dynamic.mapper..*.*(..))")
public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("执行AOP");
//获取当前要执行的函数签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
Method method = signature.getMethod();
//获取当前函数上的指定注解
TargetDataSource targetDataSource = method.getAnnotation(TargetDataSource.class);
//如果有指定注解的话设置为注解的值,没有的话设置默认腾讯云
if (targetDataSource == null) {
DynamicDataSourceManger.set(DataSourceEnum.TENCENT);
} else {
String dyType = targetDataSource.value();
switch (dyType) {
case "TENCENT":
DynamicDataSourceManger.set(DataSourceEnum.TENCENT);
break;
case "LOCAL":
DynamicDataSourceManger.set(DataSourceEnum.LOCAL);
break;
}
}
//执行函数
return joinPoint.proceed();
}
}

再次执行函数发现效果和上面是一样的。

Spring Boot系列:七、 实现Mybatis多数据源切换的更多相关文章

  1. Spring Boot (七)MyBatis代码自动生成和辅助插件

    一.简介 1.1 MyBatis Generator介绍 MyBatis Generator 是MyBatis 官方出品的一款,用来自动生成MyBatis的 mapper.dao.entity 的框架 ...

  2. Spring Boot 系列总目录

    一.Spring Boot 系列诞生原因 上学那会主要学的是 Java 和 .Net 两种语言,当时对于语言分类这事儿没什么概念,恰好在2009年毕业那会阴差阳错的先找到了 .Net 的工作,此后就开 ...

  3. Spring Boot (七): Mybatis极简配置

    Spring Boot (七): Mybatis极简配置 1. 前言 ORM 框架的目的是简化编程中的数据库操作,经过这么多年的发展,基本上活到现在的就剩下两家了,一个是宣称可以不用写 SQL 的 H ...

  4. Spring Boot系列(三):Spring Boot整合Mybatis源码解析

    一.Mybatis回顾 1.MyBatis介绍 Mybatis是一个半ORM框架,它使用简单的 XML 或注解用于配置和原始映射,将接口和Java的POJOs(普通的Java 对象)映射成数据库中的记 ...

  5. 国内最全的Spring Boot系列之二

    历史文章 <国内最全的Spring Boot系列之一> 视频&交流平台 SpringBoot视频:http://t.cn/R3QepWG Spring Cloud视频:http:/ ...

  6. 07.深入浅出 Spring Boot - 数据访问之Mybatis(附代码下载)

    MyBatis 在Spring Boot应用非常广,非常强大的一个半自动的ORM框架. 代码下载:https://github.com/Jackson0714/study-spring-boot.gi ...

  7. spring boot系列01--快速构建spring boot项目

    最近的项目用spring boot 框架 借此学习了一下 这里做一下总结记录 非常便利的一个框架 它的优缺点我就不在这背书了 想了解的可以自行度娘谷歌 说一下要写什么吧 其实还真不是很清楚,只是想记录 ...

  8. Spring Boot系列(四):Spring Boot源码解析

    一.自动装配原理 之前博文已经讲过,@SpringBootApplication继承了@EnableAutoConfiguration,该注解导入了AutoConfigurationImport Se ...

  9. spring mvc+mybatis+多数据源切换

    spring mvc+mybatis+多数据源切换,选取oracle,mysql作为例子切换数据源.oracle为默认数据源,在测试的action中,进行mysql和oracle的动态切换. web. ...

  10. spring+mybatis多数据源切换

    在实际的公司项目中,很可能会遇到一个问题就是,一个java项目,但是项目中涉及两个数据库,这两个数据库还在不同IP的机子上. 遇到这种情况的时候,我们有两个选择 1.不走spring的aop方式,直接 ...

随机推荐

  1. Python3网络学习案例三:编写web server

    1. 写在前面 这里总结的并不够详细,有时间了再进行补充. 2. 设计思路 HTTP协议是建立在TCP上的1. 建立服务器端TCP套接字(绑定ip,port),等待监听连接:listen(2. 打开浏 ...

  2. K8S 搭建 Kafka:2.13-2.6.0 和 Zookeeper:3.6.2 集群

    搭建 Kafka:2.13-2.6.0 和 Zookeeper:3.6.2 集群 一.服务版本信息: Kafka:v2.13-2.6.0 Zookeeper:v3.6.2 Kubernetes:v1. ...

  3. 832. Flipping an Image —— weekly contest 84

    Flipping an Image Given a binary matrix A, we want to flip the image horizontally, then invert it, a ...

  4. 直播软件开发如何使用FFMPEG推流并保存在本地

    最近开发了基于C#的直播软件开发推流器一直不大理想,终于在不懈努力之后研究了一点成果,这边做个笔记:本文着重在于讲解下如何使用ffmpeg进行简单的推流,看似简单几行代码没有官方的文档很吃力.并获取流 ...

  5. 【SpringCloud】05.Eureka的高可用

    1.简单情况 2.为了达到Eureka的高可用,可以多个Eureka互相注册. 3.我们需要修改两处: Eureka Client Eureka Server 3.1 Eureka Client 在C ...

  6. NO.A.0001——FIO工具使用教程

    一.FIO工具安装: 1.FIO地址: 官网地址:http://freecode.com/projects/fio/ 源码安装包:http://brick.kernel.dk/snaps/fio-2. ...

  7. 性能工具-io工具

    I/O:某网上问题通过top  iotop pidstat vmstat 工具定位出io高原因,内存不够.

  8. VS2017新建MVC+ORM中的LinqDb访问数据库项目

    1.前提概述 ORM对象关系映射(Object-Relational Mapping)是一种程序技术,用于实现面向对象编程语言里不同类型系统的数据之间的转换.从效果上说,它其实是创建了一个可在编程语言 ...

  9. 阻塞队列的take、offer、put、add的一些比较

    LinkedBlockingQueue的put,add和offer的区别 最近在学习<<Java并发编程实践>>,有很多java.util.concurrent包下的新类.Li ...

  10. HBuilderX SVN地址更改(SVN服务器IP地址变更)

    HBuilderX编辑器中无法修改SVN地址,需要手动在SVN工具中修改 修改步骤: 1.右键编辑器中的SVN项目,选择打开文件所在目录 2.目录中空白处右键,选择TortoiseSVN --> ...