MyBatis一級緩存避坑完全指南
一級緩存概念
當(dāng)我們使用Mybatis進(jìn)行數(shù)據(jù)庫的操作時(shí)候,會創(chuàng)建一個(gè)SqlSession來進(jìn)行一次數(shù)據(jù)庫的會話,會話結(jié)束則關(guān)閉SqlSession對象。那么一個(gè)SqlSession的生命周期即對應(yīng)于Mybatis的一次會話。在Mybatis的一次會話中,我們很有可能多次查詢完全相同的sql語句,如果不采取措施的話,每一次查詢都查詢一次數(shù)據(jù)庫。而一次會話時(shí)間一般都是極短的,相同Sql的查詢結(jié)果極有可能完全相同。由于查詢數(shù)據(jù)庫代價(jià)是比較大的,這會導(dǎo)致系統(tǒng)的資源浪費(fèi)。
為了解決這個(gè)問題,Mybatis對每一次會話都添加了緩存操作。這個(gè)緩存的作用域?yàn)橐淮螘捴?。緩存隨著會話(SqlSession)的創(chuàng)建而產(chǎn)生,隨著會話結(jié)束而釋放。對一次會話的查詢操作,總是先查看緩存中是否存在查詢結(jié)果,如果存在則直接取緩存中的結(jié)果,不存在則查詢數(shù)據(jù)庫。這樣的話,一次會話中的完全相同的查詢則只會查詢一次,節(jié)省了系統(tǒng)資源。
引言
MyBatis 一級緩存(MyBaits 稱其為 Local Cache)無法關(guān)閉,但是有兩種級別可選:
package org.apache.ibatis.session; /** * @author Eduardo Macarron */ public enum LocalCacheScope { SESSION, //session 級別的緩存 STATEMENT //statement 級別的緩存 }
1)session 級別的緩存
在同一個(gè) sqlSession 內(nèi),對同樣的查詢將不再查詢數(shù)據(jù)庫,直接從緩存中。
驗(yàn)證代碼:
[
public static void main(String[] args) throws IOException { InputStream inputStream = new ClassPathResource("mybatis.xml").getInputStream(); SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream); SqlSession sqlSession = sqlSessionFactory.openSession(); UserDao mapper = sqlSession.getMapper(UserDao.class); System.out.println(mapper.get(1L)); System.out.println("-------------------"); System.out.println(mapper.get(1L));}
輸出:
日志輸出可以看到,第一次查詢通過數(shù)據(jù)庫查詢,第二次則沒有,直接通過緩存讀取。
坑:這種緩存策略有一個(gè)坑,在服務(wù)集群時(shí)就會出現(xiàn)問題。
假設(shè)現(xiàn)在有一個(gè)服務(wù)集群,有兩個(gè)節(jié)點(diǎn)。
首先,兩個(gè)節(jié)點(diǎn)都進(jìn)行了同樣的查詢,兩個(gè)節(jié)點(diǎn)都有自己的一級緩存,后續(xù)同樣的查詢,兩個(gè)節(jié)點(diǎn)將不再查詢數(shù)據(jù)庫。
如果此時(shí)節(jié)點(diǎn) 1 執(zhí)行了 update 語句,那么節(jié)點(diǎn) 1 的一級緩存會被刷新,而節(jié)點(diǎn) 2 的一級緩存不會改變。
2)statement 級別的緩存
避坑: 為了避免這個(gè)問題,可以將一級緩存的級別設(shè)為 statement 級別的,這樣每次查詢結(jié)束都會清掉一級緩存。MyBatis 源碼如下:
在 MyBatis 的配置文件中,添加以下配置:
驗(yàn)證代碼和上面的一樣不變。
輸出:
可以看到,即使是同樣的查詢,每次查詢都是直接讀取數(shù)據(jù)庫了。
避坑完畢。
緩存是不可能不要緩存的,這個(gè)時(shí)候,就需要使用緩存中間件了,由緩存中間件管理緩存。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
使用SpringBoot與EasyExcel實(shí)現(xiàn)復(fù)雜的導(dǎo)入導(dǎo)出
這篇文章主要介紹了使用SpringBoot與EasyExcel實(shí)現(xiàn)復(fù)雜的導(dǎo)入導(dǎo)出,EasyExcel是一個(gè)快速解決大文件內(nèi)存溢出的Excel處理工具,它能讓你在不用考慮性能、內(nèi)存等因素的情況下,快速完成Excel的讀、寫等功能,需要的朋友可以參考下2023-10-10Springboot自定義mybatis攔截器實(shí)現(xiàn)擴(kuò)展
本文主要介紹了Springboot自定義mybatis攔截器實(shí)現(xiàn)擴(kuò)展,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12java實(shí)現(xiàn)excel導(dǎo)出合并單元格的步驟詳解
這篇文章主要介紹了java實(shí)現(xiàn)excel導(dǎo)出合并單元格,通過使用Apache POI庫,我們可以方便地創(chuàng)建Excel文件、填充數(shù)據(jù)、合并單元格和導(dǎo)出Excel文件,需要的朋友可以參考下2023-04-04教你如何監(jiān)控 Java 線程池運(yùn)行狀態(tài)的操作(必看)
這篇文章主要介紹了教你如何監(jiān)控 Java 線程池運(yùn)行狀態(tài)的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02Java分支結(jié)構(gòu)程序設(shè)計(jì)實(shí)例詳解
這篇文章主要介紹了Java分支結(jié)構(gòu)程序設(shè)計(jì)例題,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-01-01Java中獲取?List中最后一個(gè)元素3種方法以及實(shí)際應(yīng)用
這篇文章主要給大家介紹了關(guān)于Java中獲取?List中最后一個(gè)元素3種方法以及實(shí)際應(yīng)用的相關(guān)資料,由于List的索引是從0開始的,所以最后一個(gè)元素的索引是List的大小減1,需要的朋友可以參考下2023-11-11