mybatis二級緩存默認未開啟源碼的問題
說明
mybatis的二級緩存,我們通常說,默認是關(guān)閉的,這個結(jié)論是沒有問題的,但是我覺得有幾個點需要說明白
二級緩存的時候,需要有幾個配置必須開啟,二級緩存才會生效
- 1.全局配置文件中的cacheEnable屬性設(shè)置為true
- 2.mapper.xml文件中,配置節(jié)點
- 3.mapper.xml文件中,select語句的useCache配置為true
這三者同時滿足,才會使用二級緩存
上面這三個配置,在源碼中,解析了之后,分別對應(yīng)這三個類中的屬性
我們知道,mybatis中的配置文件,在解析了之后,最終會把所有解析的結(jié)果,映射到一個類中org.apache.ibatis.session.Configuration
- 1.對應(yīng)著 configuration.cacheEnabled屬性
- 2.對應(yīng)著 Configuration --> mappedStatements --> cache
- 3.對應(yīng)著 Configuration --> mappedStatements --> useCache
我們常說二級緩存默認是關(guān)閉的,并不是說這三個配置默認都沒有配置,翻了源碼之后,默認不做任何配置的情況下:
- 1.全局配置文件中的CacheEnable屬性,默認是true
- 2.mapper.xml文件中的節(jié)點如果沒有配置, 那MappedStatement對象中的cache屬性為null
- 3.如果select語句的useCache沒有配置,select語句默認是true,其他是false
所以,雖然這種也是默認關(guān)閉的,但是我覺得需要說明白,并不是二級緩存用到的這幾個屬性默認都是false
源碼
針對這三個配置,我們來看下默認的配置對應(yīng)的源碼
cacheEnable
對CacheEnable的解析是在
org.apache.ibatis.builder.xml.XMLConfigBuilder#parseConfiguration

我們接著來看賦值的邏輯這里,如果props為空(就是在全局配置文件中沒有配置settings節(jié)點),此時在給各個屬性賦值的時候,分別指定了默認值,可以看到,cacheEnable屬性值默認是true;所以這里可以證明
第一點

節(jié)點的解析
org.apache.ibatis.builder.xml.XMLMapperBuilder#configurationElement

由于我沒有配置cache節(jié)點,所以,這里為null,不會執(zhí)行下面的useNewCache方法,為了證明第二點這個說法,我們假設(shè)此時指定了cache節(jié)點,我們來看下useNewCache()方法中會做什么


可以看到,在useNewCache方法中,初始化了cache之后,會把cache賦值在當前類的一個屬性中,currentCache;為什么要關(guān)注這個屬性?
因為后面在給mappedStatement的cache屬性賦值時,會使用currentCache;
截止到這里,我們只需要知道,如果在mapper.xml文件中,沒有配置節(jié)點,那org.apache.ibatis.builder.MapperBuilderAssistant#currentCache這個屬性就是null
useCache的解析
org.apache.ibatis.builder.xml.XMLStatementBuilder#parseStatementNode
這個方法是解析了一個sql段之后的邏輯,可以看到,這里所有的屬性,都是我們在中可以設(shè)置的,其中有一個useCache屬性

看下這里的意思:
- 如果useCache這個屬性為null,就會直接返回def 就是默認值,可以看到這里的默認值是isSelect;isSelect是在上面賦值的,如果當前是select語句,就是true,如果當前是非select,就是false;
其含義是:
- 如果沒有配置useCache屬性,select語句默認使用二級緩存,其他語句不使用

這里只是給useCache賦值了,我們看下在哪里把這個賦值放到了mappedStatement對象中
builderAssistant.addMappedStatement(id, sqlSource, statementType, sqlCommandType,
fetchSize, timeout, parameterMap, parameterTypeClass, resultMap, resultTypeClass,
resultSetTypeEnum, flushCache, useCache, resultOrdered,
keyGenerator, keyProperty, keyColumn, databaseId, langDriver, resultSets);
在解析了所有的屬性之后,會調(diào)用這個方法,給mappedStatent對象賦值

在賦值這里,有兩個重要的代碼段
第一個是使用useCache賦值,這里使用的就是前面入?yún)⒌?;使用currentCache給cache屬性賦值,我們前面有說過,如果在mapper.xml文件中,沒有配置節(jié)點,那此時的currentCache就是null
第二個:是把當前的mappedStatement對象添加到configuration的mappedStatements屬性中
使用二級緩存源碼
所以以上可以證明這三個配置對應(yīng)的默認值,我們再來看下,sql在真正執(zhí)行時,是如何使用二級緩存的
上面有說過,默認的CacheEnable屬性默認是true,所以在初始化executor對象的時候,默認會初始化CachingExecutor對象,所以,無論是否使用了二級緩存,都會先調(diào)用到cachingExecutor對象中
可以看到,在真正調(diào)用二級緩存前,會有兩層判斷,分別是cache和useCache;這兩個配置,cache默認是null,useCache默認是true,默認情況下,第一個if判斷都無法進去,就會直接走一級緩存查詢

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
java如何實現(xiàn)自動生成數(shù)據(jù)庫設(shè)計文檔
以前我們還需要手寫數(shù)據(jù)庫設(shè)計文檔、現(xiàn)在可以通過引入screw核心包來實現(xiàn)Java?數(shù)據(jù)庫文檔一鍵生成。本文將具體介紹一下如何通過java自動生成數(shù)據(jù)庫設(shè)計文檔,需要的朋友可以參考下2021-11-11
Java實戰(zhàn)之實現(xiàn)一個好用的MybatisPlus代碼生成器
這篇文章主要介紹了Java實戰(zhàn)之實現(xiàn)一個好用的MybatisPlus代碼生成器,文中有非常詳細的代碼示例,對正在學習java的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-04-04
java數(shù)據(jù)結(jié)構(gòu)算法稀疏數(shù)組示例詳解
這篇文章主要為大家介紹了java數(shù)據(jù)結(jié)構(gòu)算法稀疏數(shù)組示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-06-06

