為什么mybatis中的SqlSession一定要關(guān)閉
mybatis的SqlSession一定要關(guān)閉
今天在使用mybatis查詢(xún)數(shù)據(jù)時(shí),出現(xiàn)了一個(gè)很奇怪的問(wèn)題。同一條sql語(yǔ)句,查詢(xún)時(shí)快時(shí)慢,并且有一定的規(guī)律性,大概每10次查詢(xún)中有一次會(huì)特別特別的慢,快的只需要1ms,慢的要20000ms,sql代碼及快慢時(shí)間截圖如下:
select fknr from jq_fkqk where jjxh = ?
快的情況

慢的情況

通過(guò)日志打印mybatis查詢(xún)信息時(shí),我觀察到特別慢的時(shí)候,并不是因?yàn)樗樵?xún)很慢,而是因?yàn)樗枰却欢魏荛L(zhǎng)的時(shí)間才開(kāi)始:
==> Preparing: select fknr from jq_fkqk where jjxh = ?
說(shuō)明時(shí)間長(zhǎng)就出現(xiàn)在這里,這個(gè)等待的時(shí)間。
??
為什么會(huì)要等待這么久呢?我自然而然的看了一下上一次查詢(xún)的函數(shù),發(fā)現(xiàn)在函數(shù)里面沒(méi)有session.close(),把這一句加上,問(wèn)題就解決了,速度就飛快了。
??
我本來(lái)想查看官方文檔,查看原因,但是沒(méi)找到,只能結(jié)合我自己的理解來(lái)分析一下。SqlSession是通過(guò)SqlSessionFactory來(lái)構(gòu)造的,相當(dāng)于維護(hù)一個(gè)連接池,當(dāng)我們不停的進(jìn)行查詢(xún)的時(shí)候,由于沒(méi)有關(guān)閉連接,導(dǎo)致與數(shù)據(jù)庫(kù)的連接數(shù)量達(dá)到了一個(gè)上限(可能連接池有最大連接數(shù),但是我們有找到文檔)。
到達(dá)上限之后,再次請(qǐng)求查詢(xún)時(shí),F(xiàn)actory說(shuō)沒(méi)有連接了,讓你先等一下,它先去判斷哪些SqlSession已經(jīng)沒(méi)有人使用了(類(lèi)似于垃圾回收機(jī)制),然后調(diào)用相應(yīng)的進(jìn)程去自動(dòng)關(guān)閉沒(méi)用的session連接,注意調(diào)用進(jìn)程可是要排隊(duì)的,也要耗時(shí)間。
等關(guān)閉了沒(méi)有用的session之后,F(xiàn)actory通知你,有空閑的session了,開(kāi)始準(zhǔn)備你的查詢(xún)吧,所有才會(huì)有等待很長(zhǎng)一段時(shí)間才出現(xiàn):
==> Preparing: select fknr from jq_fkqk where jjxh = ?
這純屬我自己的理解,但是重點(diǎn)還是表達(dá)出來(lái)了,就是Mybatis中的session一定要手動(dòng)去關(guān)閉它,session.close(),不然會(huì)占著資源,導(dǎo)致性能下降?。。?/p>
mybatis中SqlSession使用事項(xiàng)
一.SqlSession的使用范圍
SqlSession中存儲(chǔ)的是編譯好的sql語(yǔ)句,這些sql語(yǔ)句是mybatis配置文件讀取mapper.xml文件生成的,將sql語(yǔ)句存儲(chǔ)到SqlSessionFactory和SqlSession中。封裝了對(duì)數(shù)據(jù)庫(kù)的操作,如:查詢(xún)、插入、更新、刪除等。
通過(guò)SqlSessionFactory創(chuàng)建SqlSession,而SqlSessionFactory是通過(guò)SqlSessionFactoryBuilder加載配置文件進(jìn)行創(chuàng)建。在開(kāi)發(fā)中SqlSession在每次使用完都要進(jìn)行關(guān)閉,使用時(shí)創(chuàng)建,也就是多例的,線程安全。SqlSessionFactory在整個(gè)類(lèi)中只有一個(gè)對(duì)象,也就是單例的,單例的線程不安全。
二.SqlSessionFactoryBuilder
SqlSessionFactoryBuilder用于創(chuàng)建SqlSessionFacoty,SqlSessionFacoty一旦創(chuàng)建完成就不需要SqlSessionFactoryBuilder了,因?yàn)镾qlSession是通過(guò)SqlSessionFactory生產(chǎn),所以可以將SqlSessionFactoryBuilder當(dāng)成一個(gè)工具類(lèi)使用,最佳使用范圍是方法范圍即方法體內(nèi)局部變量。
三.SqlSessionFactory
SqlSessionFactory是一個(gè)接口,接口中定義了openSession的不同重載方法,SqlSessionFactory的最佳使用范圍是整個(gè)應(yīng)用運(yùn)行期間,一旦創(chuàng)建后可以重復(fù)使用,通常以單例模式管理SqlSessionFactory。
四.SqlSession
SqlSession是一個(gè)面向用戶的接口, sqlSession中定義了數(shù)據(jù)庫(kù)操作,默認(rèn)使用DefaultSqlSession實(shí)現(xiàn)類(lèi)。
執(zhí)行過(guò)程如下:
1、加載數(shù)據(jù)源等配置信息
Environment environment = configuration.getEnvironment();
2、創(chuàng)建數(shù)據(jù)庫(kù)鏈接
3、創(chuàng)建事務(wù)對(duì)象
4、創(chuàng)建Executor,SqlSession所有操作都是通過(guò)Executor完成,mybatis源碼如下:
if (ExecutorType.BATCH == executorType) {
executor = newBatchExecutor(this, transaction);
} elseif (ExecutorType.REUSE == executorType) {
executor = new ReuseExecutor(this, transaction);
} else {
executor = new SimpleExecutor(this, transaction);
}
if (cacheEnabled) {
executor = new CachingExecutor(executor, autoCommit);
}
SqlSession的實(shí)現(xiàn)類(lèi)即DefaultSqlSession,此對(duì)象中對(duì)操作數(shù)據(jù)庫(kù)實(shí)質(zhì)上用的是Executor
小結(jié)一下
每個(gè)線程都應(yīng)該有它自己的SqlSession實(shí)例。SqlSession的實(shí)例不能共享使用,它也是線程不安全的。因此最佳的范圍是請(qǐng)求或方法范圍。絕對(duì)不能將SqlSession實(shí)例的引用放在一個(gè)類(lèi)的靜態(tài)字段或?qū)嵗侄沃小?/p>
打開(kāi)一個(gè) SqlSession;使用完畢就要關(guān)閉它。通常把這個(gè)關(guān)閉操作放到 finally 塊中以確保每次都能執(zhí)行關(guān)閉。如下:
SqlSession session = sqlSessionFactory.openSession();
try {
// do work
} finally {
session.close();
}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
IntelliJ IDEA全局內(nèi)容搜索和替換教程圖解
很多朋友在做項(xiàng)目時(shí),會(huì)在整個(gè)項(xiàng)目里活指定文件夾下進(jìn)行全局搜索和替換,下面小編給大家?guī)?lái)了IntelliJ IDEA全局內(nèi)容搜索和替換教程圖解,需要的朋友參考下吧2018-04-04
動(dòng)態(tài)配置Spring Boot日志級(jí)別的全步驟
這篇文章主要給大家介紹了關(guān)于動(dòng)態(tài)配置Spring Boot日志級(jí)別的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring Boot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
Java利用反射實(shí)現(xiàn)框架類(lèi)的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于Java利用反射實(shí)現(xiàn)框架類(lèi)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
SpringBoot中多環(huán)境啟動(dòng)配置的教程詳解
在SpringBoot項(xiàng)目的生命周期中,存在不同的環(huán)境,我們就需要針對(duì)不同環(huán)境制定不同名稱(chēng)的配置文件,里面放置不同環(huán)境下所需的配置項(xiàng),下面小編就來(lái)和大家詳細(xì)講講SpringBoot如何進(jìn)行多環(huán)境啟動(dòng)配置的吧2024-02-02
JAVA中的日期LocalDate類(lèi)詳細(xì)用法講解
Java中存在一個(gè)日歷表示法的類(lèi)庫(kù),LocalDate類(lèi),如果只想要指定的年月日就可以用這個(gè)LocalDate類(lèi),下面這篇文章主要給大家介紹了關(guān)于JAVA中日期LocalDate類(lèi)詳細(xì)用法講解的相關(guān)資料,需要的朋友可以參考下2024-01-01

