SpringBoot自定义注解实现多数据源

前置学习

需要了解 注解、Aop、SpringBoot整合Mybatis的使用。

数据准备

基础项目代码:https://gitee.com/J_look/spring-boot-all-demo

数据库SQL 项目中有提供,修改基本信息即可

行动起来

添加依赖

利用 AOP 可以实现对某些代码的解耦,不需要硬编码编写。

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-aop</artifactId>
  4. </dependency>

开启AOP支持

也可以像图中一样,也开启事务管理器,下文会演示事务失效的问题。

定义枚举

这里定义的枚举,代表我们不同的数据库。

  1. public enum DataSourceType {
  2. MYSQL_DATASOURCE1,
  3. MYSQL_DATASOURCE2,
  4. }

定义数据源管理器

由于本人实力原因,解答不了大家这里的疑惑。大致功能 通过修改本地线程的值,来实现数据源的切换

  1. @Component
  2. @Primary
  3. public class DataSourceManagement extends AbstractRoutingDataSource {
  4. public static ThreadLocal<String> flag = new ThreadLocal<>();
  5. /**
  6. * 注入数据源
  7. */
  8. @Resource
  9. private DataSource mysqlDataSource1;
  10. /**
  11. * 注入数据源
  12. */
  13. @Resource
  14. private DataSource mysqlDataSource2;
  15. public DataSourceManagement() {
  16. flag.set(DataSourceType.MYSQL_DATASOURCE1.name());
  17. }
  18. @Override
  19. protected Object determineCurrentLookupKey() {
  20. return flag.get();
  21. }
  22. @Override
  23. public void afterPropertiesSet() {
  24. Map<Object, Object> targetDataSource = new ConcurrentHashMap<>();
  25. targetDataSource.put(DataSourceType.MYSQL_DATASOURCE1.name(), mysqlDataSource1);
  26. targetDataSource.put(DataSourceType.MYSQL_DATASOURCE2.name(), mysqlDataSource2);
  27. // 设置数据源来源
  28. super.setTargetDataSources(targetDataSource);
  29. // 设置默认数据源
  30. super.setDefaultTargetDataSource(mysqlDataSource1);
  31. super.afterPropertiesSet();
  32. }
  33. }

自定义注解

@target:表示自定义注解能用在哪里

  1. @Target({ElementType.TYPE,ElementType.METHOD})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Documented
  4. public @interface TargetDataSource {
  5. DataSourceType value() default DataSourceType.MYSQL_DATASOURCE1;
  6. }

定义切面类

可以看到,我们的切面类采用的是环绕通知,博主在写这篇文章之前,做过大量测试,也参考过比较多的文章,总结以下几点:

  • 采用前置通知,虽然能实现数据源的切换,但是会导致事务失效。(推荐视频)
  • 采用环绕通知,事务不会失效,但是切换数据源却实现不了。(由Bean的加载顺序导致的,下文中的@Order就可以解决,@Transactional默认级别是最后加载。可以查看日志信息知晓。)
  1. @Component
  2. @Aspect
  3. @Slf4j
  4. @Order(Ordered.LOWEST_PRECEDENCE-1) // Bean加载顺序
  5. public class TargetDataSourceAspect {
  6. @Around("@within(TargetDataSource) || @annotation(TargetDataSource)")
  7. public Object beforeNoticeUpdateDataSource(ProceedingJoinPoint joinPoint) {
  8. TargetDataSource annotation = null;
  9. Class<? extends Object> target = joinPoint.getTarget().getClass();
  10. if (target.isAnnotationPresent(TargetDataSource.class)) {
  11. // 判断类上是否标注着注解
  12. annotation = target.getAnnotation(TargetDataSource.class);
  13. log.info("类上标注了注解");
  14. } else {
  15. Method method = ((MethodSignature) joinPoint.getSignature()).getMethod();
  16. if (method.isAnnotationPresent(TargetDataSource.class)) {
  17. // 判断方法上是否标注着注解,如果类和方法上都没有标注,则报错
  18. annotation = method.getAnnotation(TargetDataSource.class);
  19. log.info("方法上标注了注解");
  20. } else {
  21. throw new RuntimeException("@TargetDataSource注解只能用于类或者方法上, 错误出现在:[" +
  22. target.toString() + " " + method.toString() + "];");
  23. }
  24. }
  25. // 切换数据源
  26. DataSourceManagement.flag.set(annotation.value().name());
  27. Object result = null;
  28. try {
  29. // 执行目标代码
  30. result = joinPoint.proceed();
  31. } catch (Throwable e) {
  32. e.printStackTrace();
  33. }
  34. return result;
  35. }
  36. }

Service

启动测试类:

  1. import look.word.datasource.service.BookService;
  2. import org.junit.jupiter.api.Test;
  3. import org.springframework.boot.test.context.SpringBootTest;
  4. import javax.annotation.Resource;
  5. /**
  6. * @author : look-word
  7. * 2022-10-10 23:11
  8. **/
  9. @SpringBootTest
  10. public class TestBookMapper {
  11. @Resource
  12. private BookService bookService;
  13. @Test
  14. void updatePrice() {
  15. System.out.println(bookService.updatePrice(1, 777));
  16. }
  17. }

观察执行日志可知,数据源的切换实现了,事务也没有失效。

参考文章:

SpringBoot 自定义注解 实现多数据源的更多相关文章

  1. [技术博客] SPRINGBOOT自定义注解

    SPRINGBOOT自定义注解 在springboot中,有各种各样的注解,这些注解能够简化我们的配置,提高开发效率.一般来说,springboot提供的注解已经佷丰富了,但如果我们想针对某个特定情景 ...

  2. java/springboot自定义注解实现AOP

    java注解 即是注释了,百度解释:也叫元数据.一种代码级别的说明. 个人理解:就是内容可以被代码理解的注释,一般是一个类. 元数据 也叫元注解,是放在被定义的一个注解类的前面 ,是对注解一种限制. ...

  3. SpringBoot自定义注解

    1.注解的概念 注解是一种能被添加到java代码中的元数据,类.方法.变量.参数和包都可以用注解来修饰.注解对于它所修饰的代码并没有直接的影响. 2.注解的使用范围 1)为编译器提供信息:注解能被编译 ...

  4. 使用IDEA创建SpringBoot自定义注解

    创建SpringBoot项目 添加组织名 选择web 输入项目名称 创建后目录结构为 使用Spring的AOP先加入Maven依赖 <dependency> <groupId> ...

  5. springboot+自定义注解实现灵活的切面配置

    利用aop我们可以实现业务代码与系统级服务例如日志记录.事务及安全相关业务的解耦,使我们的业务代码更加干净整洁. 最近在做数据权限方面的东西,考虑使用切面对用户访问进行拦截,进而确认用户是否对当前数据 ...

  6. SpringBoot自定义注解、AOP打印日志

    前言 在SpringBoot中使用自定义注解.aop切面打印web请求日志.主要是想把controller的每个request请求日志收集起来,调用接口.执行时间.返回值这几个重要的信息存储到数据库里 ...

  7. SpringBoot 自定义注解

    新增注解类 NotRepeatSubmit.java package com.example.demo.annotation; import java.lang.annotation.ElementT ...

  8. SpringBoot自定义注解+异步+观察者模式实现业务日志保存

    一.前言 我们在企业级的开发中,必不可少的是对日志的记录,实现有很多种方式,常见的就是基于AOP+注解进行保存,但是考虑到程序的流畅和效率,我们可以使用异步进行保存,小编最近在spring和sprin ...

  9. SpringBoot自定义注解@YamlPropertySource加载yml或者yaml文件(扩展了@PropertySource)

    1:概述 SpringBoot的@PropertySource注解只支持加载 properties结尾的文件.当使用@ConfigurationProperties 注解配合@EnableConfig ...

随机推荐

  1. python开发环境配置(Windows)

    简介 由于在搭建pyhon开发环境时会出现各种各样的问题,因此将这些问题记录下来 1.下载python 从官网下载对应系统的python版本(最新稳定版即可):官网地址为:python下载地址, 建议 ...

  2. B+树索引页大小是如何确定的?

    B+树简介 在正式介绍本文的主题前,需要对 B+ 树有一定的了解,B+树是一种磁盘上数据的索引结构,大概长这个样子. B+树的叶子节点是所有的数据,非叶子节点称为索引页,索引页里有若干个索引项,本例中 ...

  3. Tomcat报错:类XXXServlet不是Servlet 解决方法

    学习servlet 结果对应网页打不开,报错 HTTP状态 500 - 内部服务器错误 类型 异常报告 消息 类HelloServlet不是Servlet ... 根本原因. java.lang.Cl ...

  4. JS 取Json数据中对象特定属性值

    解析JSON JSON 数据 var str = '[{"a": "1","b": "2"}, {"a&quo ...

  5. Excel 逻辑函数(一):IF 和 IFS

    IF IF 函数有三个参数,第一个为条件判断,第二个是当条件为真时执行的表达式,第三个是条件为假时执行的表达式. IF(A1="是", A2 * 0.8, 0),如果 A1 单元格 ...

  6. z—libirary最新地址获取,zlibirary地址获取方式,zliabary最新地址,zliabary官网登录方式,zliabary最新登陆

    Z-Library(缩写为z-lib,以前称为BookFinder)是Library Genesis的镜像,一个影子图书馆项目,用于对学术期刊文章.学术文本和大众感兴趣的书籍(其中一些是盗版的)进行文 ...

  7. 硬件IIC驱动原理

    1.IIC物理层 IIC通信属于同步半双工通信,IIC总线由两根信号线组成.一根是数据线SDA,一根是时钟线SCL,时钟线只能由主机发送给从机,数据线可以双向进行通信,总线上可挂载多个设备,挂载数量受 ...

  8. C#基础_C#计算样本标准差和总体标准差

    首先我们先了解样本标准差和总体标准差: 样本标准差=方差的算术平方根=s=sqrt(((x1-x)^2 +(x2-x)^2 +......(xn-x)^2)/(n-1)) 总体标准差=σ=sqrt(( ...

  9. 【IDEA】IDEA怎么汉化&汉化后怎么转回英文

    ① 英文转中文 1.点击左上角的File,然后选择Setting 2.达到Setting页面选择Plugins 3.在搜索框搜索chinese,选择中文语言包下载 4.找到下载插件,选择勾选上,然后o ...

  10. Html飞机大战(八):子弹的移动和管理

    好家伙,这应该是这个小游戏最难的几个点之一了 现在我们要做出子弹射击的效果我们应该如何处理? 1.首先我们要确定几个变量和方法的关系 变量: 子弹  bullet  弹夹(用来装子弹的东西)bulle ...