专栏首页码猿技术专栏SpringBoot整合多数据源的巨坑

SpringBoot整合多数据源的巨坑

导读

  • 前面文章最后留了几个问题供大家思考,今天一一揭晓。

配置如何优化

  • 上文整合的过程中的还顺带整合Mybatis和TransactionManager,为什么还要重新定义他们呢?SpringBoot不是给我们都配置好了吗?注意,此处优化就是这两个配置去掉,直接用SpringBoot的自动配置,顿时高级了,别人一看你的代码如此简单就实现了多数据源的切换,牛叉不?
  • 如何去掉?SpringBoot万变不离自动配置类,且看MybatisAutoConfiguration,如下:

123456

@org.springframework.context.annotation.Configuration@ConditionalOnClass({ SqlSessionFactory.class, SqlSessionFactoryBean.class })@ConditionalOnSingleCandidate(DataSource.class)@EnableConfigurationProperties(MybatisProperties.class)@AutoConfigureAfter(DataSourceAutoConfiguration.class)public class MybatisAutoConfiguration implements InitializingBean {

  • 不多帖了,都是废话,看前几行就行了,醒目的一行啊,@ConditionalOnSingleCandidate(DataSource.class),什么鬼?该注解的意思就是IOC容器中只有一个指定的候选对象才起作用,但是我们注入了几个DataSource,足足三个啊,这还起作用吗?那不废话嘛。
  • 事务管理器也是一样,且看DataSourceTransactionManagerAutoConfiguration,如下:

12345

public class DataSourceTransactionManagerAutoConfiguration { @Configuration @ConditionalOnSingleCandidate(DataSource.class) static class DataSourceTransactionManagerConfiguration {

  • 又看到了什么,@ConditionalOnSingleCandidate(DataSource.class)同样的醒目,mmp,这不玩我呢吗。这怎么搞?
  • 咦,不着急,此时就要看看@ConditionalOnSingleCandidate注解搞了什么,进去看看,有如下的介绍:

1

The condition will also match if multiple matching bean instances are already contained in the BeanFactory but a primary candidate has been defined; essentially, the condition match if auto-wiring a bean with the defined type will succeed.

  • 什么鬼,看不懂,英语太差了吧,不着急,陈某给大家推荐一个IDEA插件,文档翻译更加专注于程序员的专业术语,不像xx度翻译,如下:
  • 好了,翻译准确了就知道了,大致意思就是IOC容器中允许你有多个候选对象,但是你必须有一个主(primary)候选对象,顿时灵光一现,这不就是@Primary注解吗,艹,我这也太优秀了吧。
  • 二话不说,直接开撸,轻轻松松一个注解搞定,此时的数据源配置变得简单多了,如下:

12345678910111213141516171819202122232425262728293031323334353637383940414243444546

/** * @Description 数据源的配置 * @Author CJB * @Date 2020/3/9 13:45 */@Configuration@MapperScan(basePackages = {"com.vivachek.service.dao","com.vivachek.service.dao2"})public class DatasourceConfig { /** * 注入数据源1 */ @ConfigurationProperties(prefix = "spring.datasource1") @Bean(value = "dataSource1") public DataSource dataSource1() { return new DruidDataSource(); } /** * 第二个数据源 */ @Bean(name = "dataSource2") @ConfigurationProperties(prefix = "spring.datasource2") public DataSource dataSource2() { return new DruidDataSource(); } /** * 动态数据源 * * @return */ @Bean @Primary public DynamicDataSource dynamicDataSource() { DynamicDataSource dataSource = new DynamicDataSource(); //默认数据源,在没有切换数据源的时候使用该数据源 dataSource.setDefaultTargetDataSource(dataSource2()); HashMap<Object, Object> map = Maps.newHashMap(); map.put("dataSource1", dataSource1()); map.put("dataSource2", dataSource2()); //设置数据源Map,动态切换就是根据key从map中获取 dataSource.setTargetDataSources(map); return dataSource; }}

  • 直接在DynamicDataSource添加了一个@Primary就省去了SqlSessionFactory和TransactionManager的手动配置,是不是很easy并且显得自己很牛叉,太有成就感了…..
  • 好了,牛也吹了,运行一下吧,满怀期待等待30秒…….,what?什么鬼?失败了,抛出了异常,如下:
  • 什么鬼,循环依赖异常,搞什么飞机,一万个草泥马在奔腾在横无际涯的草原上。。。。。。。。
  • 别急,还有后续,关注我,将会定时更新后续文章。另外需要源码的联系我,微信联系方式在个人独立博客【关于我】中,加我注明来意,谢谢。
  • 别忘了点赞哟,多来走动走动呗……….

动态路由数据源添加@Primary报循环依赖异常

  • 前面文章Spring解决循环依赖有说过Spring对于循环依赖是完全能够解决的,没有读过的小伙伴建议看一下,里面详细的讲述了Spring是如何解决循环依赖的,此处就不再赘述了。
  • 既然Spring能够解决循环依赖,为什么这里又会报循环依赖的异常呢?我们不妨跟着代码看看是怎样的循环依赖,如下:
  • 上面两个数据源都是自己定义的,先不用看,那么肯定是DataSourceInitializerInvoker造成的循环依赖了,果不其然,其中确实依赖了DataSource,源码如下:

123456

DataSourceInitializerInvoker(ObjectProvider<DataSource> dataSource, DataSourceProperties properties, ApplicationContext applicationContext) { this.dataSource = dataSource; this.properties = properties; this.applicationContext = applicationContext; }

  • what?即使依赖了又怎样?Spring不是可以解决循环依赖吗?别着急下面分析
  • ObjectProvider应该不陌生吧,其实内部就是从IOC容器中获取Bean而已,但是,转折来了……… ,这是什么,这是构造器,Spring能解决构造器的循环依赖吗?答案是不能,所以原因找到了,这里不再细说了,欲知原因请读Spring解循环依赖
  • 问题找到了,如何解决?此时心中一万个草泥马奔腾,怎么解决呢?
  • 哈哈,此时插播一条广告,本人的独立博客已经发布了很多文章,感兴趣的可以收藏一下,【关于我】中有我的微信联系方式,欢迎交流。
  • 回到正题,如何解决?很简单,找到这个DataSourceInitializerInvoker是什么时候注入到IOC容器中的,因此我们找到了DataSourceAutoConfiguration,继而找到了DataSourceInitializationConfiguration这个配置类,源码如下:

12345678910111213141516171819202122232425262728

@Configuration@ConditionalOnClass({ DataSource.class, EmbeddedDatabaseType.class })@EnableConfigurationProperties(DataSourceProperties.class)@Import({ DataSourcePoolMetadataProvidersConfiguration.class, DataSourceInitializationConfiguration.class })public class DataSourceAutoConfiguration { @Configuration @Conditional(EmbeddedDatabaseCondition.class) @ConditionalOnMissingBean({ DataSource.class, XADataSource.class }) @Import(EmbeddedDataSourceConfiguration.class) protected static class EmbeddedDatabaseConfiguration { } @Configuration @Conditional(PooledDataSourceCondition.class) @ConditionalOnMissingBean({ DataSource.class, XADataSource.class }) @Import({ DataSourceConfiguration.Hikari.class, DataSourceConfiguration.Tomcat.class, DataSourceConfiguration.Dbcp2.class, DataSourceConfiguration.Generic.class, DataSourceJmxConfiguration.class }) protected static class PooledDataSourceConfiguration { } } @Configuration@Import({ DataSourceInitializerInvoker.class, DataSourceInitializationConfiguration.Registrar.class })class DataSourceInitializationConfiguration {

  • 贴了那么多代码谁看的懂?草泥马又奔腾了,可以看到源码中出现了两次@ConditionalOnMissingBean({ DataSource.class, XADataSource.class }),这什么鬼,不多说了,相信读过SpringBoot源码的都知道,这个配置类根本不起作用啊,那还要它干嘛,直接搞掉不就完事了。好了,分析到这里终于知道解决的方案了,搞掉DataSourceAutoConfiguration,怎么搞呢?一个注解搞定。

12

//排除配置类@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})

  • 问题迎刃而解了,简单不,惊喜不,不好,又奔腾了。。。。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • springboot整合多数据源

    贪挽懒月
  • SpringBoot之SpringBoot整合多数据源

    本来按照视屏来说,到上一章的打包运行就没有了,但是我百度翻了好一大波,找到了他的上一期中的其他剩余视屏

    彼岸舞
  • SpringBoot整合MongoDB多数据源

    GET http://localhost:8080/test/login?username=Zhangsan

    wo.
  • SpringBoot实战(三):整合Mybatis配置多数据源

    最近接到一个新需求,经过分析后做了相应的设计;其中需要在一个项目中操做不同的数据源;于是进行了相关验证;在此记录一下验证过程。

    用户2781897
  • Springboot 整合Mybatis多数据源并动态切换

    来源:https://juejin.im/post/5d8705e65188253f4b629f47

    JAVA葵花宝典
  • SpringBoot 整合mybatis 多数据源以及分库分表

    说实话,这章本来不打算讲的,因为配置多数据源的网上有很多类似的教程。但是最近因为项目要用到分库分表,所以让我研究一下看怎么实现。我想着上一篇博客讲了多环境的配置...

    程序员爱酸奶
  • SpringBoot之解决整合多数据源多事物注解问题

      因为整合了多数据源,就代表我的项目中存在多个事物管理器,这样就不能直接使用@Transactional

    彼岸舞
  • SpringBoot如何整合多个数据源,看这篇就够了

    SpringBoot现在是很多很多公司应用的后端框架,因为它搭建快,能更好、更快速的整合其他第三方。那么随着业务的不断扩展,业务量的增加,这时候就会牵扯到分库分...

    一个程序员的成长
  • SpringBoot整合数据源的驼峰命名问题

    上次跟大家分享了SpringBoot如何配置多个数据源,配置多个数据源是成功了,但是有个小问题,就是关于驼峰命名的字段并没有查询出来,也就是说,我们的驼峰命名规...

    一个程序员的成长
  • 第二十四章:SpringBoot项目整合JPA多数据源配置

    恒宇少年
  • SpringBoot之解决整合多数据源分布式事物问题

      上一章只是解决了单事物问题,也就是说同时只能使用自己的数据源,并指定事物管理,才能使用,那么如果同时使用多个数据源,就会产生分布式事物问题

    彼岸舞
  • springboot之整合druid并配置数据源监控

    这里使用druid的jar,当然springboot也有其自己的druid启动器:

    西西嘛呦
  • Spring Boot 如何整合多个数据源?

    SpringBoot现在是很多很多公司应用的后端框架,因为它搭建快,能更好、更快速的整合其他第三方。那么随着业务的不断扩展,业务量的增加,这时候就会牵扯到分库分...

    Java_老男孩
  • [SpringBoot实战]快速配置多数据源(整合MyBatis)

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明...

    蛮三刀酱
  • Spring boot 整合dynamic实现多数据源

    Jacob丶
  • 极简Spring Boot整合MyBatis多数据源

    关于多数据源的配置,前面和大伙介绍过JdbcTemplate多数据源配置,那个比较简单,本文来和大伙说说MyBatis多数据源的配置。 其实关于多数据源,我的态...

    江南一点雨
  • [SpringBoot]快速配置多数据源(整合MyBatis)

    由于业务需求,需要同时在SpringBoot中配置两套数据源(连接两个数据库),要求能做到service层在调用各数据库表的mapper时能够自动切换数据源,也...

    Rude3Knife的公众号
  • SpringBoot 的多数据源配置

    最近在项目开发中,需要为一个使用 MySQL 数据库的 SpringBoot 项目,新添加一个 PLSQL 数据库数据源,那么就需要进行 SpringBoot ...

    未读代码
  • 【程序源代码】SpringBoot整合JDBC数据库连接池

    关键字:本篇为SpringBoot整合JDBC数据库教程,内容比较简单,比较适合小白学习。

    程序源代码

扫码关注云+社区

领取腾讯云代金券