Java?MyBatis本地緩存原理詳解
背景
出現(xiàn)了一次生產(chǎn)事故,事情是這樣的,我們有一個(gè)項(xiàng)目,Java訪問(wèn)數(shù)據(jù)庫(kù)的框架使用的是MyBatis。然后一個(gè)業(yè)務(wù)員在系統(tǒng)中查詢(xún)了一個(gè)訂單,發(fā)現(xiàn)這個(gè)訂單是未支付的狀態(tài),于是業(yè)務(wù)員聯(lián)系客戶(hù),讓客戶(hù)支付,客戶(hù)支付完成后,業(yè)務(wù)員又去系統(tǒng)查詢(xún),結(jié)果還是未支付狀態(tài),刷新了頁(yè)面也是一樣,不過(guò)過(guò)了一會(huì)就好了。業(yè)務(wù)員把這個(gè)延遲問(wèn)題,反饋給了我們。我就看代碼,只是一個(gè)簡(jiǎn)單的select * from order where id = ?
語(yǔ)句調(diào)用。這個(gè)時(shí)候就想到了我們今天故事的主角,MyBatis的緩存機(jī)制。
發(fā)現(xiàn)問(wèn)題
復(fù)現(xiàn)
public class Main { public static void main(String[] args) throws IOException, InterruptedException { SqlSessionFactory build = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml")); UsersMapper mapper = build.openSession().getMapper(UsersMapper.class); List<Users> users = mapper.selectAll(); System.out.println(JSONUtil.toJsonStr(users)); // 在這睡眠期間去使用update語(yǔ)句修改數(shù)據(jù)庫(kù)信息。 Thread.sleep(1000 * 10); List<Users> users1 = mapper.selectAll(); System.out.println(JSONUtil.toJsonStr(users1)); } }
看到上面的問(wèn)題,很自然的就想到之前面試時(shí)候背的八股文,MyBatis的一二級(jí)緩存。
解決問(wèn)題
其實(shí)這個(gè)問(wèn)題,對(duì)于實(shí)效性不強(qiáng)的項(xiàng)目的話,完全可以和業(yè)務(wù)說(shuō),網(wǎng)絡(luò)延遲,等一等就可以了。 如果說(shuō)實(shí)用性比較強(qiáng)的項(xiàng)目,可以選擇禁用這個(gè)緩存。不過(guò)這樣也會(huì)帶來(lái)一個(gè)問(wèn)題,也就是沒(méi)有緩存后,很多查詢(xún),查詢(xún)結(jié)果相同的SQL語(yǔ)句,原本只需要執(zhí)行一遍就可以,這里會(huì)請(qǐng)求很多次。增高DB的IO負(fù)擔(dān)。
探究緩存的原理
Sql查詢(xún)部分深入
在上一篇Java MyBatis是如何執(zhí)行一條SQL語(yǔ)句的講MyBatis的文章中已經(jīng)說(shuō)到了,mapper會(huì)經(jīng)過(guò)動(dòng)態(tài)代理去執(zhí)行SqlSession的query方法。
閱讀緩存部分源碼,需要跟隨查詢(xún)方法往下追著看。
接著,我們這里可以看到,一個(gè)Switch有很多的case,很顯而易見(jiàn),我們會(huì)進(jìn)入Select中,隨后Select的代碼塊中,又有很多個(gè)If判斷,因?yàn)槲覀兊姆椒ǚ祷氐氖且粋€(gè)List,正好就命中了returnsMany方法。
最終調(diào)用SqlSession中的Select方法,到達(dá)這里接著往里追。
到達(dá)執(zhí)行器處理SQL語(yǔ)句的這一塊了,可見(jiàn)封裝了很多層,再往下追。
初見(jiàn)緩存
看到這里,終于見(jiàn)到緩存相關(guān)字樣了,這里去CreateCacheKey,看到這里我突然想起了,剛工作時(shí)候的一次面試,有個(gè)傻*面試官,問(wèn)我說(shuō)MyBatis的緩存Key是怎么生成,當(dāng)時(shí)我真想給他兩耳巴。繼續(xù)摳下面的query方法,可以看到把上面生成的key作為參數(shù)傳了下來(lái)。
追到這里就可以看到一個(gè)比較關(guān)鍵的代碼了,從localCache中調(diào)用了GetObject方法
這里看到下面的方法,從這里可以證明,MyBatis默認(rèn)就是用本地緩存,所以寫(xiě)代碼的時(shí)候自然也要記得處理緩存不一致的問(wèn)題。
告一段落
到這里這一篇文章就結(jié)束了。一般來(lái)說(shuō),作為Java工程師,在工作中還是經(jīng)常用到MyBatis這個(gè)點(diǎn)的,如果你在面試中真的問(wèn)到了這么一道緩存題。而那么巧你就知道,那么郎有情,妾有意,恭喜您,點(diǎn)亮漲薪1K的成就。希望大家在當(dāng)前大環(huán)境這么不好的情況下,多多學(xué)習(xí),多多面試,提高核心競(jìng)爭(zhēng)力,等大環(huán)境回暖各個(gè)年入50萬(wàn)+。
番外篇-Myabtis創(chuàng)建CacheKey的算法。
回到這一行代碼,可以看到這個(gè)方法返回的是一個(gè)CacheKey的JavaClass,先不急著看這個(gè)方法的具體實(shí)現(xiàn),先去看下這個(gè)類(lèi)的構(gòu)成,和構(gòu)造方法。
構(gòu)造方法
這個(gè)CacheKey類(lèi)一共有兩個(gè)構(gòu)造方法,可以看到的在有參構(gòu)造方法中,調(diào)用了無(wú)參方法,隨后調(diào)用了,updateAll方法,能在構(gòu)造中被調(diào)用的一般都是比較重要的,一會(huì)來(lái)看一下這個(gè)方法的實(shí)現(xiàn),先來(lái)看下無(wú)參構(gòu)造中的幾個(gè)變量。
hashcode、multiplier、cout、updateList,值得注意的是hashCode和multiplier兩個(gè)成員變量,都給賦值了初始值。
這兩個(gè)變量記憶一下,然后去看updateAll方法。
這個(gè)方法比較簡(jiǎn)單,遍歷了objects入?yún)?,傳入update方法,繼續(xù)去追update方法。
看到這里可以看出這里大概是計(jì)算HashCode的一個(gè)地方。只是最后會(huì)把算過(guò)的值放入U(xiǎn)pdateList中。
隨后去看這個(gè)類(lèi)的tostring方法。
這里也就是具體的方法了
根據(jù)冒號(hào)分割,hashCode:checkSum:updateList的每個(gè)元素
再回過(guò)頭來(lái)看一下CreateCacheKey方法看下最后生成是什么
最后答案留給評(píng)論互動(dòng)吧。
結(jié)束語(yǔ)
到此這篇關(guān)于Java MyBatis本地緩存原理詳解的文章就介紹到這了,更多相關(guān)Java MyBatis本地緩存 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)現(xiàn)登錄密碼強(qiáng)度校驗(yàn)的項(xiàng)目實(shí)踐
本文主要介紹了Java實(shí)現(xiàn)登錄密碼強(qiáng)度校驗(yàn)的項(xiàng)目實(shí)踐,包括使用正則表達(dá)式匹配校驗(yàn)和密碼強(qiáng)度校驗(yàn)工具類(lèi)這兩種方法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01IDEA設(shè)置JVM運(yùn)行參數(shù)的方法步驟
這篇文章主要介紹了IDEA設(shè)置JVM運(yùn)行參數(shù)的方法步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08JAVA實(shí)現(xiàn)多線程的兩種方法實(shí)例分享
這篇文章介紹了JAVA實(shí)現(xiàn)多線程的兩種方法實(shí)例分享,有需要的朋友可以參考一下2013-08-08詳解Spring注入集合(數(shù)組、List、Map、Set)類(lèi)型屬性
這篇文章主要介紹了詳解Spring注入集合(數(shù)組、List、Map、Set)類(lèi)型屬性,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01Java內(nèi)存模型之重排序的相關(guān)知識(shí)總結(jié)
重排序是指編譯器和處理器為了優(yōu)化性能而對(duì)指令序列進(jìn)行重新排序的一種手段,文中詳細(xì)介紹了Java重排序的相關(guān)知識(shí),需要的朋友可以參考下2021-06-06spring事務(wù)@Transactional失效原因及解決辦法小結(jié)
今天就跟大家聊聊有關(guān)spring中@Transactional失效原因及解決辦法小結(jié),主要從三個(gè)方面考慮,具有一定的參考價(jià)值,感興趣的可以了解一下2023-08-08OpenJDK源碼解析之System.out.println詳解
這篇文章主要介紹了OpenJDK源碼解析之System.out.println詳解,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-04-04SpringBoot+mybatis實(shí)現(xiàn)多數(shù)據(jù)源支持操作
這篇文章主要介紹了SpringBoot+mybatis實(shí)現(xiàn)多數(shù)據(jù)源支持操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-10-10