mybatis一級(jí)緩存和二級(jí)緩存的區(qū)別及說明
我們通常說mybatis中一級(jí)緩存是sqlSession級(jí)別的,二級(jí)緩存是namespace級(jí)別的,這篇筆記主要來記錄下這么說的原理
結(jié)論
先說結(jié)論吧,一級(jí)緩存之所以說是sqlSession級(jí)別的,是因?yàn)橐患?jí)緩存的數(shù)據(jù)是存放在了sqlSession的一個(gè)內(nèi)部屬性中,所以,每次openSession()開啟一個(gè)sqlSession之后,一級(jí)緩存就會(huì)失效
二級(jí)緩存之所以可以跨sqlSession,是因?yàn)槎?jí)緩存的數(shù)據(jù),是存放在mappedStatement對(duì)象中的一個(gè)內(nèi)部屬性中,這里說的內(nèi)部屬性其實(shí)不太準(zhǔn)確,但是先忽略,后面會(huì)詳細(xì)解釋
我們知道,mybatis在啟動(dòng)的時(shí)候,會(huì)解析全局配置文件,會(huì)把mapper.xml文件中的一個(gè)個(gè)sql片段,解析成一個(gè)個(gè)mappedStatement對(duì)象,所以,這里二級(jí)緩存自然也就是namespace級(jí)別的
源碼
在mybatis源碼中,CachingExecutor是二級(jí)緩存的處理類,BaseExecutor是一級(jí)緩存的處理類
我們先來看一級(jí)緩存的處理
一級(jí)緩存
這里是一級(jí)緩存的處理,會(huì)發(fā)現(xiàn),這里是從localCache中根據(jù)key獲取value的,換而言之,這里的localCache大概率就是我們所謂的一級(jí)緩存了,我們看下localCache是在哪里賦值的
可以看到,localCache是在BaseExecutor的構(gòu)造函數(shù)中,每次new 出來的,所以,這里我們可以知道,只要new 一個(gè)BaseExecutor對(duì)象,就會(huì)有一個(gè)localCache,那對(duì)應(yīng)的一級(jí)緩存就不一樣,我們?cè)倏聪?,在正常的sql執(zhí)行過程中,BaseExecutor是在什么時(shí)候初始化的
org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSession()
org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSessionFromDataSource
org.apache.ibatis.session.Configuration#newExecutor()
這里可以看到,在通過openSession()初始化sqlSession對(duì)象的時(shí)候,就會(huì)初始化一個(gè)executor對(duì)象
所以,我們說一級(jí)緩存是executor級(jí)別的
二級(jí)緩存
我們接著來看二級(jí)緩存的邏輯
二級(jí)緩存是從這里的tcm這個(gè)對(duì)象中來獲取的,但是實(shí)際上,我們調(diào)到tcm.getObject()方法中,會(huì)發(fā)現(xiàn)實(shí)際上是從入?yún)⒌腸ache來獲取的,所以我們需要看下這里入?yún)⒌腸ache是從哪里獲取到的
可以看到這個(gè)方法的第一行,是從MappedStatement對(duì)象中獲取到的
可以看到,mappedStatement對(duì)象是從configuration對(duì)象中獲取到,所以我們要看下configuration的mappedStatement對(duì)象是什么時(shí)候賦值的
org.apache.ibatis.builder.MapperBuilderAssistant#addMappedStatement()
在這個(gè)方法中,可以看到,最終會(huì)把statement對(duì)象設(shè)置的configuration中,同時(shí)需要注意的是,我們用到的cache對(duì)象是從currentCache來的,所以我們需要關(guān)注,currentCache是從哪里來的?
在同類中,org.apache.ibatis.builder.MapperBuilderAssistant#useNewCache
這個(gè)方法中,會(huì)給currentCache賦值
如果看過前面兩篇關(guān)于mybatis源碼解析博客的,應(yīng)該會(huì)比較清楚,這里的useNewCache()方法,就是在解析mapper.xml文件中的節(jié)點(diǎn)時(shí),調(diào)用的
org.apache.ibatis.builder.xml.XMLMapperBuilder#configurationElement
org.apache.ibatis.builder.xml.XMLMapperBuilder#cacheElement
org.apache.ibatis.builder.MapperBuilderAssistant#useNewCache
截圖中的這個(gè)方法是解析一個(gè)mapper.xml的邏輯
所以,我們發(fā)現(xiàn)
在解析mapper.xml文件的時(shí)候,會(huì)解析配置的節(jié)點(diǎn),在解析這個(gè)節(jié)點(diǎn)的時(shí)候,會(huì)初始化一個(gè)cache對(duì)象,這個(gè)cache對(duì)象,會(huì)賦值給currentCache然后繼續(xù)解析一個(gè)個(gè)sql片段,這一個(gè)mapper.xml文件中,無論有多少個(gè)sql,這里的currentCache都是一樣的二級(jí)緩存在使用的時(shí)候,會(huì)從mappedStatement對(duì)象中獲取currentCache,如果currentCache不為null,就表示需要使用二級(jí)緩存然后從currentCache,根據(jù)key獲取value這里需要知道的是:這里的currentCache,也是對(duì)PerpetualCache的包裝,以前文章有介紹過,這次不做過多的解讀
所以,我們說二級(jí)緩存是namespace級(jí)別的
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- 一文搞懂MyBatis一級(jí)緩存和二級(jí)緩存
- Java Mybatis一級(jí)緩存和二級(jí)緩存
- MyBatis一級(jí)緩存與二級(jí)緩存原理與作用分析
- Mybatis詳細(xì)對(duì)比一級(jí)緩存與二級(jí)緩存
- Mybatis?一級(jí)緩存和二級(jí)緩存原理區(qū)別
- Mybatis的一級(jí)緩存和二級(jí)緩存原理分析與使用
- 關(guān)于mybatis的一級(jí)緩存和二級(jí)緩存的那些事兒
- Mybatis 一級(jí)緩存與二級(jí)緩存的實(shí)現(xiàn)
- MyBatis 延遲加載、一級(jí)緩存、二級(jí)緩存(詳解)
- MyBatis中一級(jí)緩存和二級(jí)緩存的區(qū)別
相關(guān)文章
SpringBoot Mybatis動(dòng)態(tài)數(shù)據(jù)源切換方案實(shí)現(xiàn)過程
這篇文章主要介紹了SpringBoot+Mybatis實(shí)現(xiàn)動(dòng)態(tài)數(shù)據(jù)源切換方案過程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04Java導(dǎo)出excel時(shí)合并同一列中相同內(nèi)容的行思路詳解
這篇文章主要介紹了Java導(dǎo)出excel時(shí)合并同一列中相同內(nèi)容的行,需要的朋友可以參考下2018-06-06關(guān)于SpringMVC的數(shù)據(jù)綁定@InitBinder注解的使用
這篇文章主要介紹了關(guān)于SpringMVC的數(shù)據(jù)綁定@InitBinder注解的使用,在SpringMVC中,數(shù)據(jù)綁定的工作是由 DataBinder 類完成的,DataBinder可以將HTTP請(qǐng)求中的數(shù)據(jù)綁定到Java對(duì)象中,需要的朋友可以參考下2023-07-07Java中通過三級(jí)緩存解決Spring循環(huán)依賴詳解
這篇文章主要介紹了Java中通過三級(jí)緩存解決Spring循環(huán)依賴詳解,當(dāng)出現(xiàn)兩個(gè)或多個(gè) Bean 在初始化時(shí)相互依賴的情況時(shí),Spring Boot 會(huì)將其中一個(gè) Bean 提前暴露出來,以便其他 Bean 能夠在初始化時(shí)正確地引用它,這一策略能有效避免循環(huán)依賴導(dǎo)致的問題,需要的朋友可以參考下2023-09-09java中Integer包裝類裝箱的一個(gè)細(xì)節(jié)詳解
Java中的Integer是int的包裝類型,下面這篇文章主要給大家介紹了關(guān)于java中Integer包裝類裝箱的一個(gè)細(xì)節(jié)的相關(guān)資料,文中介紹的這個(gè)細(xì)節(jié)挺重要的,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起看看吧2018-07-07java 中ThreadLocal本地線程和同步機(jī)制的比較
這篇文章主要介紹了java 中ThreadLocal本地線程和同步機(jī)制的比較的相關(guān)資料,需要的朋友可以參考下2017-03-03Java Synchronize下的volatile關(guān)鍵字詳解
這篇文章主要介紹了Java Synchronize下的volatile關(guān)鍵字詳解,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03