资讯

精准传达 • 有效沟通

从品牌网站建设到网络营销策划,从策略到执行的一站式服务

mybatis多数据源踩坑,数据库连接经常断开问题-创新互联

问题

最近某项目上出现一个奇怪的问题,就是数据库经常隔几小时就报连接已关闭

创新互联是一家集网站建设,义乌企业网站建设,义乌品牌网站建设,网站定制,义乌网站建设报价,网络营销,网络优化,义乌网站推广为一体的创新建站企业,帮助传统企业提升企业形象加强企业竞争力。可充分满足这一群体相比中小企业更为丰富、高端、多元的互联网需求。同时我们时刻保持专业、时尚、前沿,时刻以成就客户成长自我,坚持不断学习、思考、沉淀、净化自己,让我们为更多的企业打造出实用型网站。

mybatis多数据源踩坑,数据库连接经常断开问题

mybatis多数据源踩坑,数据库连接经常断开问题

即使是加了如下配置也依然不行,网上也没找到什么文章解释这个坑

test-on-borrow: true
test-while-idle: true
validation-query: select 1 from dual

排查

网上查不到,那就只能自己推敲猜测了。因为是mybatis多数据源的配置,所以每个db我都有专门写一个config作为连接配置。

mybatis多数据源踩坑,数据库连接经常断开问题

看着DataSourceConfig的代码,我突然想到,会不会是因为我使用到是DataSource默认创建方法,所以并没有读取到我写在application.yml的配置:

 @Bean(name = "db1DataSource")
 @ConfigurationProperties(prefix = "spring.datasource.db1")
 @Primary
 public DataSource dbDataSource() {
 return DataSourceBuilder.create().build();
 }

果断跟进build()方法

	public DataSource build() {
		Class type = getType();
		DataSource result = BeanUtils.instantiate(type);
		maybeGetDriverClassName();
		bind(result);
		return result;
	}

打个断点可以看到此时返回的result的是一个全新的DataSource

mybatis多数据源踩坑,数据库连接经常断开问题

所以我们可以通过修改dbDataSource()方法,写入我们的配置参数:

 @Value("${spring.datasource.db1.url}")
 private String url;
 @Value("${spring.datasource.db1.username}")
 private String username;
 @Value("${spring.datasource.db1.password}")
 private String password;
 @Value("${spring.datasource.db1.tomcat.test-on-borrow}")
 private boolean testOnBorrow;
 @Value("${spring.datasource.db1.tomcat.test-while-idle}")
 private boolean testWhileIdle;
 @Value("${spring.datasource.db1.tomcat.validation-query}")
 private String validationQuery;
 @Value("${spring.datasource.db1.tomcat.max-idle}")
 private int maxIdle;
 @Value("${spring.datasource.db1.tomcat.min-idle}")
 private int minIdle;
 @Value("${spring.datasource.db1.tomcat.initial-size}")
 private int initialSize;
 @Value("${spring.datasource.db1.tomcat.max-active}")
 private int maxActive;
 @Value("${spring.datasource.db1.tomcat.time-between-eviction-runs-millis}")
 private int timeBetweenEvictionRunsMillis;
 @Bean(name = "db1DataSource")
 @ConfigurationProperties(prefix = "spring.datasource.db1")
 @Primary
 public DataSource dbDataSource() {
 org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource();
 dataSource.setDriverClassName("oracle.jdbc.driver.OracleDriver");
 dataSource.setUrl(url);
 dataSource.setUsername(username);
 dataSource.setPassword(password);
 dataSource.setMaxActive(maxActive);
 dataSource.setMinIdle(minIdle);
 dataSource.setMaxIdle(maxIdle);
 dataSource.setTestOnBorrow(testOnBorrow);
 dataSource.setTestWhileIdle(testWhileIdle);
 dataSource.setValidationQuery(validationQuery);
 dataSource.setTimeBetweenEvictionRunsMillis(timeBetweenEvictionRunsMillis);
 dataSource.setInitialSize(initialSize);
 return dataSource;
 //return DataSourceBuilder.create().build();
 }

二次排查

通过上面方法确实可以解决问题了,但是我突然想到,默认创建的DataSource是没有url,username,password等必要的基础信息的。那这几个配置参数是为什么又可以写入进去呢?

这个时候我看到了我们dbDataSource方法上有一个@Bean(name = "db1DataSource"),于是大胆猜测我们这些配置参数的注入是第一次创建的时候通过Spring的IOC注入的。通过我的Debug发现事实也确实如此。

对DataBinder类的bind方法打断点,

	public void bind(PropertyValues pvs) {
		MutablePropertyValues mpvs = (pvs instanceof MutablePropertyValues) ?
				(MutablePropertyValues) pvs : new MutablePropertyValues(pvs);
		doBind(mpvs);
	}

mybatis多数据源踩坑,数据库连接经常断开问题

我们可以看到方法的调用路径

mybatis多数据源踩坑,数据库连接经常断开问题

看到了我们熟悉的refresh大法,这一部分Spring源码相关请看Spring源码分析

此时只剩最后一个疑惑了,我们的url,username,password等既然能通过IOC注入到DataSource里,那为什么其他的参数不可以呢?我随着DataSource类一路往上,找到他的父类接口PoolConfiguration,看到了所有的参数和getset方法。

mybatis多数据源踩坑,数据库连接经常断开问题

再看一眼我的application.yml配置文件里的参数

spring:
 datasource:
 db1:
 url: 
 username: 
 password: 
 driver-class-name: oracle.jdbc.driver.OracleDriver
 tomcat:
 max-wait: 10000
 max-active: 30
 test-on-borrow: true
 max-idle: 5
 db2:
 xxx
 ....

终于找到这个坑了!

原来Spring data默认使用tomcat-jdbc的连接池的时候,配置的参数是

spring:
 datasource:
 url: 
 username: 
 password: 
 driver-class-name: oracle.jdbc.driver.OracleDriver
 tomcat:
 max-wait: 10000
 max-active: 30
 test-on-borrow: true
 max-idle: 5

而当使用多数据源配置的时候,简单的以为只是复制过去即可,所以Spring IOC注入的时候,读的到的tomcat.max-wait并不能匹配到DataSource里的setMaxWait方法。自然就不起作用了。

所以这个问题只需要将配置文件改为如下即可

spring:
 datasource:
 db1:
 url: 
 username: 
 password: 
 driver-class-name: oracle.jdbc.driver.OracleDriver
 max-wait: 10000
 max-active: 30
 test-on-borrow: true
 test-while-idle: true
 validation-query: select 1 from dual
 max-idle: 5
 db2:
 xxx
 ....

总结

这个问题从结果上来看,那真是简单的不行,但是从过程上来说,不仅让我又复习了一遍spring IOC的流程,也让我感觉到这种一步一步解剖问题,把多个知识点连接起来的成就感。如果之前没有学习spring的源码,我这次大概率也不会想到去看bean的注入吧

点击获取 附送学习进阶架构资料、PDF书籍文档、面试资料

mybatis多数据源踩坑,数据库连接经常断开问题

mybatis多数据源踩坑,数据库连接经常断开问题

另外有需要云服务器可以了解下创新互联scvps.cn,海内外云服务器15元起步,三天无理由+7*72小时售后在线,公司持有idc许可证,提供“云服务器、裸金属服务器、高防服务器、香港服务器、美国服务器、虚拟主机、免备案服务器”等云主机租用服务以及企业上云的综合解决方案,具有“安全稳定、简单易用、服务可用性高、性价比高”等特点与优势,专为企业上云打造定制,能够满足用户丰富、多元化的应用场景需求。


网站栏目:mybatis多数据源踩坑,数据库连接经常断开问题-创新互联
新闻来源:http://cdkjz.cn/article/cejjic.html
多年建站经验

多一份参考,总有益处

联系快上网,免费获得专属《策划方案》及报价

咨询相关问题或预约面谈,可以通过以下方式与我们联系

大客户专线   成都:13518219792   座机:028-86922220