MyBatis中一級(jí)緩存和二級(jí)緩存的區(qū)別
MyBatis 提供了兩級(jí)緩存機(jī)制:一級(jí)緩存(Session級(jí)別)和二級(jí)緩存(全局級(jí)別),以提高應(yīng)用的性能通過減少數(shù)據(jù)庫的查詢次數(shù)。
一級(jí)緩存(Session級(jí)別)
一級(jí)緩存是基于 SQL 會(huì)話(SqlSession
)的,它是默認(rèn)開啟的。一級(jí)緩存的生命周期與 SQL 會(huì)話一致,當(dāng)會(huì)話結(jié)束時(shí),緩存也隨之消失。這意味著,在同一個(gè) SQL 會(huì)話中,對(duì)于相同的查詢請(qǐng)求,第一次會(huì)從數(shù)據(jù)庫中獲取數(shù)據(jù),并緩存結(jié)果;之后相同的查詢請(qǐng)求將直接從緩存中獲取數(shù)據(jù),不會(huì)再去訪問數(shù)據(jù)庫。
try (SqlSession session = sqlSessionFactory.openSession()) { BlogMapper mapper = session.getMapper(BlogMapper.class); // 第一次查詢,數(shù)據(jù)來自數(shù)據(jù)庫 Blog blog1 = mapper.selectBlogById(1); // 第二次查詢,相同的會(huì)話中,數(shù)據(jù)來自一級(jí)緩存 Blog blog2 = mapper.selectBlogById(1); }
一級(jí)緩存主要是利用了一個(gè) Map
來存儲(chǔ)緩存數(shù)據(jù),關(guān)鍵代碼在 DefaultSqlSession
類中的 selectList
方法里,它使用了 Executor
的 query
方法來實(shí)現(xiàn)。
public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) { try { MappedStatement ms = configuration.getMappedStatement(statement); return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER); } catch (Exception e) { throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e); } finally { ErrorContext.instance().reset(); } }
在 CachingExecutor
類的 query
方法中實(shí)現(xiàn)了一級(jí)緩存的邏輯:
public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException { BoundSql boundSql = ms.getBoundSql(parameter); CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql); return query(ms, parameter, rowBounds, resultHandler, key, boundSql); }
二級(jí)緩存(全局級(jí)別)
二級(jí)緩存是跨 SQL 會(huì)話的,它基于 namespace 級(jí)別。開啟二級(jí)緩存后,數(shù)據(jù)會(huì)存儲(chǔ)在全局作用域內(nèi),這意味著,即使 SQL 會(huì)話關(guān)閉,緩存數(shù)據(jù)仍然可用,可以被其他 SQL 會(huì)話復(fù)用。
要啟用二級(jí)緩存,需要在 MyBatis 配置文件中添加相應(yīng)的配置,并在映射文件中明確指定哪些映射器使用二級(jí)緩存。
<!-- 在 mybatis-config.xml 中啟用全局二級(jí)緩存 --> <configuration> <settings> <setting name="cacheEnabled" value="true"/> </settings> </configuration>
在映射文件中使用二級(jí)緩存:
<!-- 在 Mapper.xml 中開啟二級(jí)緩存 --> <cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
二級(jí)緩存的關(guān)鍵實(shí)現(xiàn)在 CachingExecutor
類:
public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException { Cache cache = ms.getCache(); if (cache != null) { flushCacheIfRequired(ms); if (ms.isUseCache() && resultHandler == null) { ensureNoOutParams(ms, boundSql); @SuppressWarnings("unchecked") List<E> list = (List<E>) tcm.getObject(cache, key); if (list == null) { list = delegate.query(ms, parameter, rowBounds, resultHandler, key, boundSql); tcm.putObject(cache, key, list); // issue #578 and #116 } return list; } } return delegate.query(ms, parameter, rowBounds, resultHandler, key, boundSql); }
區(qū)別總結(jié)
- 作用范圍:一級(jí)緩存是基于
SqlSession
的,其生命周期也隨之綁定;而二級(jí)緩存是基于 Mapper 的 namespace,對(duì)整個(gè)應(yīng)用有效。 - 生命周期:一級(jí)緩存隨著 SQL 會(huì)話的結(jié)束而失效,二級(jí)緩存則可以跨會(huì)話使用。
- 可自定義性:二級(jí)緩存提供了更多的自定義設(shè)置,例如失效策略、大小限制等,通過在 Mapper.xml 中配置
<cache>
標(biāo)簽來實(shí)現(xiàn)。 - 數(shù)據(jù)安全:二級(jí)緩存由于是跨會(huì)話的,使用時(shí)需要更加注意數(shù)據(jù)的一致性問題。在使用二級(jí)緩
到此這篇關(guān)于MyBatis中一級(jí)緩存和二級(jí)緩存的區(qū)別的文章就介紹到這了,更多相關(guān)MyBatis 一級(jí)緩存和二級(jí)緩存內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot實(shí)現(xiàn)分布式驗(yàn)證碼登錄方案小結(jié)
驗(yàn)證碼登錄作為一種有效的防護(hù)手段,可以防止惡意gongji、暴力pojie等,本文主要介紹了SpringBoot實(shí)現(xiàn)分布式驗(yàn)證碼登錄方案小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2024-12-12使用Springboot實(shí)現(xiàn)OAuth服務(wù)的示例詳解
OAuth(Open Authorization)是一個(gè)開放標(biāo)準(zhǔn),用于授權(quán)第三方應(yīng)用程序訪問用戶資源,而不需要共享用戶憑證。本文主要介紹了如何使用Springboot實(shí)現(xiàn)一個(gè)OAuth服務(wù),需要的可以參考一下2023-05-05通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法
今天小編就為大家分享一篇關(guān)于通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-03-03Java中利用BitMap位圖實(shí)現(xiàn)海量級(jí)數(shù)據(jù)去重
有許多方法可以用來去重,比如使用列表、集合等等,但這些方法通常只適用于一般情況,然而,當(dāng)涉及到大量數(shù)據(jù)去重時(shí),常見的 Java Set、List,甚至是 Java 8 的新特性 Stream 流等方式就顯得不太合適了,本文給大家介紹了Java中利用BitMap位圖實(shí)現(xiàn)海量級(jí)數(shù)據(jù)去重2024-04-04Java?Web實(shí)戰(zhàn)之使用三層架構(gòu)與Servlet構(gòu)建登錄注冊(cè)模塊
這篇文章介紹了如何使用三層架構(gòu)(View,?Service,?DAO)和JDBCTemplate技術(shù)在JavaWeb環(huán)境下實(shí)現(xiàn)登錄和注冊(cè)功能,詳細(xì)說明了構(gòu)建項(xiàng)目的步驟,包括創(chuàng)建數(shù)據(jù)庫表、實(shí)體類、DAO層、Service層、Servlet處理及頁面設(shè)計(jì),需要的朋友可以參考下2024-10-10