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