最近某項(xiàng)目上出現(xiàn)一個(gè)奇怪的問(wèn)題,就是數(shù)據(jù)庫(kù)經(jīng)常隔幾小時(shí)就報(bào)連接已關(guān)閉
創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),義烏企業(yè)網(wǎng)站建設(shè),義烏品牌網(wǎng)站建設(shè),網(wǎng)站定制,義烏網(wǎng)站建設(shè)報(bào)價(jià),網(wǎng)絡(luò)營(yíng)銷,網(wǎng)絡(luò)優(yōu)化,義烏網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競(jìng)爭(zhēng)力。可充分滿足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時(shí)我們時(shí)刻保持專業(yè)、時(shí)尚、前沿,時(shí)刻以成就客戶成長(zhǎng)自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。即使是加了如下配置也依然不行,網(wǎng)上也沒(méi)找到什么文章解釋這個(gè)坑
test-on-borrow:?true test-while-idle:?true validation-query:?select?1?from?dual
網(wǎng)上查不到,那就只能自己推敲猜測(cè)了。因?yàn)槭莔ybatis多數(shù)據(jù)源的配置,所以每個(gè)db我都有專門寫一個(gè)config作為連接配置。
看著DataSourceConfig的代碼,我突然想到,會(huì)不會(huì)是因?yàn)槲沂褂玫绞荄ataSource默認(rèn)創(chuàng)建方法,所以并沒(méi)有讀取到我寫在application.yml的配置:
?@Bean(name?=?"db1DataSource") ?@ConfigurationProperties(prefix?=?"spring.datasource.db1") ?@Primary ?public?DataSource?dbDataSource()?{ ?return?DataSourceBuilder.create().build(); ?}
果斷跟進(jìn)build()方法
public?DataSource?build()?{ Class<??extends?DataSource>?type?=?getType(); DataSource?result?=?BeanUtils.instantiate(type); maybeGetDriverClassName(); bind(result); return?result; }
打個(gè)斷點(diǎn)可以看到此時(shí)返回的result的是一個(gè)全新的DataSource
所以我們可以通過(guò)修改dbDataSource()方法,寫入我們的配置參數(shù):
?@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(); ?}
通過(guò)上面方法確實(shí)可以解決問(wèn)題了,但是我突然想到,默認(rèn)創(chuàng)建的DataSource是沒(méi)有url,username,password等必要的基礎(chǔ)信息的。那這幾個(gè)配置參數(shù)是為什么又可以寫入進(jìn)去呢?
這個(gè)時(shí)候我看到了我們dbDataSource方法上有一個(gè)@Bean(name = "db1DataSource"),于是大膽猜測(cè)我們這些配置參數(shù)的注入是第一次創(chuàng)建的時(shí)候通過(guò)Spring的IOC注入的。通過(guò)我的Debug發(fā)現(xiàn)事實(shí)也確實(shí)如此。
對(duì)DataBinder類的bind方法打斷點(diǎn),
public?void?bind(PropertyValues?pvs)?{ MutablePropertyValues?mpvs?=?(pvs?instanceof?MutablePropertyValues)?? (MutablePropertyValues)?pvs?:?new?MutablePropertyValues(pvs); doBind(mpvs); }
我們可以看到方法的調(diào)用路徑
看到了我們熟悉的refresh大法,這一部分Spring源碼相關(guān)請(qǐng)看Spring源碼分析
此時(shí)只剩最后一個(gè)疑惑了,我們的url,username,password等既然能通過(guò)IOC注入到DataSource里,那為什么其他的參數(shù)不可以呢?我隨著DataSource類一路往上,找到他的父類接口PoolConfiguration,看到了所有的參數(shù)和getset方法。
再看一眼我的application.yml配置文件里的參數(shù)
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 ?....
終于找到這個(gè)坑了!
原來(lái)Spring data默認(rèn)使用tomcat-jdbc的連接池的時(shí)候,配置的參數(shù)是
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
而當(dāng)使用多數(shù)據(jù)源配置的時(shí)候,簡(jiǎn)單的以為只是復(fù)制過(guò)去即可,所以Spring IOC注入的時(shí)候,讀的到的tomcat.max-wait并不能匹配到DataSource里的setMaxWait方法。自然就不起作用了。
所以這個(gè)問(wèn)題只需要將配置文件改為如下即可
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 ?....
這個(gè)問(wèn)題從結(jié)果上來(lái)看,那真是簡(jiǎn)單的不行,但是從過(guò)程上來(lái)說(shuō),不僅讓我又復(fù)習(xí)了一遍spring IOC的流程,也讓我感覺(jué)到這種一步一步解剖問(wèn)題,把多個(gè)知識(shí)點(diǎn)連接起來(lái)的成就感。如果之前沒(méi)有學(xué)習(xí)spring的源碼,我這次大概率也不會(huì)想到去看bean的注入吧
點(diǎn)擊獲取?附送學(xué)習(xí)進(jìn)階架構(gòu)資料、PDF書籍文檔、面試資料
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。
文章名稱:mybatis多數(shù)據(jù)源踩坑,數(shù)據(jù)庫(kù)連接經(jīng)常斷開問(wèn)題-創(chuàng)新互聯(lián)
網(wǎng)站鏈接:http://m.newbst.com/article2/cejjic.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站維護(hù)、手機(jī)網(wǎng)站建設(shè)、網(wǎng)站制作、網(wǎng)站改版、標(biāo)簽優(yōu)化、面包屑導(dǎo)航
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容