mybatis中的一級緩存深入剖析
mybatis中提供有一級緩存 和 二級緩存,這里記錄一下一級緩存
一級緩存(mybatis中默認(rèn)開啟)
SqlSession級別的緩存,操作數(shù)據(jù)庫時(shí)需要構(gòu)造SQLSession對象, 在對象中有一個(gè)數(shù)據(jù)結(jié)構(gòu)(HashMap)用于存儲緩存數(shù)據(jù),不同的SQLSession對象之間的緩存數(shù)據(jù)是不共享的,即獨(dú)立的
根據(jù)第一點(diǎn),簡單一點(diǎn)講就是一級緩存是屬于對象的(個(gè)人記法)
從別處搞來一個(gè)圖,便于我們理解:
下面用spring整合mybatis來測試一下mybatis的一級緩存:
1、下面是service層實(shí)現(xiàn), 可以看到,我兩次查詢了同一個(gè)數(shù)據(jù),理論上由于mybatis中默認(rèn)開啟一級緩存, 那么第二次肯定時(shí)要從緩存中獲取,而不是創(chuàng)建SqlSession對象重新從數(shù)據(jù)庫獲取
@Autowiredprivate LsjmUserMapper lsjmUserMapper; @Overridepublic LsjmUser getUser() { // 第一次查詢 LsjmUser user = lsjmUserMapper.getUserByName('300'); System.out.println(user.toString()); // 第二次查詢 LsjmUser user1 = lsjmUserMapper.getUserByName('300'); System.out.println(user1.toString()); return user;}
前臺頁面觸發(fā)這個(gè)service后,控制臺打印:
從日志信息可以很明顯的看到,代碼中的兩次查詢構(gòu)建了兩個(gè)SqlSession對象,也就是說第二次查詢并沒有從前一次的SqlSession緩存中獲取,而是自己新建一個(gè)SQLSession對象,重新查詢,,看似,mybatis的一級緩存失效了?
2、spring 中 結(jié)合 mybatis中,默認(rèn)情況下,數(shù)據(jù)庫處于自動提交模式,每一條sql語句處于一個(gè)單獨(dú)的事務(wù)中,語句執(zhí)行完畢時(shí),如果執(zhí)行成功則隱式提交事務(wù)。而mybatis的一級緩存在這種情況下是無效的,想要一級緩存起作用,則要開啟事務(wù):
開啟事務(wù): spring使用ThreadLocal獲取當(dāng)前資源綁定同一個(gè)SQLSession
未開啟事務(wù):每次查詢,spring關(guān)閉舊的SslSession,創(chuàng)建一個(gè)新的Sqlsession對象,一級緩存補(bǔ)氣作用
下面Service層中的代碼同樣對同一個(gè)數(shù)據(jù)查詢了兩次,這次開啟了事務(wù)管理
@Autowiredprivate LsjmUserMapper lsjmUserMapper; @Override@Transactional // 開啟事務(wù)控制,當(dāng)前,spring配置文件中得先配置好public LsjmUser getUser() { // 第一次查詢 LsjmUser user = lsjmUserMapper.getUserByName('300'); System.out.println(user.toString()); // 第二次查詢 LsjmUser user1 = lsjmUserMapper.getUserByName('300'); System.out.println(user1.toString()); return user;}
前臺頁面觸發(fā)Service后:控制臺打印日志:
可以看出來第一次查詢時(shí),構(gòu)造了一個(gè)SqlSession對象,從數(shù)據(jù)庫查詢數(shù)據(jù),然后將查詢的結(jié)果存儲到一級緩存SqlSession中,第二次查詢時(shí),直接Fetched SqlSession,而不是再重新建一個(gè),此時(shí)就是從緩存中直接取數(shù)據(jù)了
開啟事務(wù)后取如何取到同一個(gè)SqlSession,源碼解析可以參考這里:spring結(jié)合mybatis時(shí)一級緩存失效問題
有一點(diǎn)需要注意:
如果對某個(gè)SqlSession執(zhí)行了commit,則會清空這個(gè)SqlSession中的一級緩存,目的是為了防止臟讀,在上面的例子中,如果第一次查詢后,做了一次更新操作,然后sqlSession.commit()了,此時(shí)會清空整個(gè)SqlSession的一級緩存,那么第二次查詢時(shí)就得重新的從數(shù)據(jù)中查詢了。
以上這篇mybatis中的一級緩存深入剖析就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. DBeaver連接MySQL的超詳細(xì)步驟2. MySQL基本調(diào)度策略淺析3. Oracle數(shù)據(jù)庫9i和10g環(huán)境下使用*.ora4. golang實(shí)現(xiàn)mysql數(shù)據(jù)庫事務(wù)的提交與回滾5. 解析MySQL binlog6. Oracle 數(shù)據(jù)字典7. 學(xué)習(xí)SQL SERVER的存儲過程-之一認(rèn)識存儲過程語法8. 關(guān)于MySQL中explain工具的使用9. mybatis-plus如何使用sql的date_format()函數(shù)查詢數(shù)據(jù)10. 高并發(fā)狀態(tài)下Replace Into造成的死鎖問題解決
