MyBatis 緩存詳解

參考文檔:MyBatis官方文檔

MyBatis的緩存主要分爲兩種一級緩存也叫本地緩存(local cache)和二級緩存(second level cache)。

一級緩存、本地緩存

一級緩存是session級緩存,即緩存只在session範圍生效。

每當一個新 session 被創建,MyBatis 就會創建一個與之相關聯的本地緩存。任何在 session 執行過的查詢語句本身都會被保存在本地緩存中,那麼,相同的查詢語句和相同的參數所產生的更改就不會二度影響數據庫了。本地緩存會被增刪改、提交事務、關閉事務以及關閉 session 所清空。

默認情況下,本地緩存數據可在整個 session 的週期內使用,這一緩存需要被用來解決循環引用錯誤和加快重複嵌套查詢的速度,所以它不可以被禁用掉,但是你可以設置 localCacheScope=STATEMENT 表示緩存僅在語句執行時有效。

注意,如果 localCacheScope 被設置爲 SESSION,那麼 MyBatis 所返回的引用將傳遞給保存在本地緩存裏的相同對象。對返回的對象(例如 list)做出任何更新將會影響本地緩存的內容,進而影響存活在 session 生命週期中的緩存所返回的值。因此,不要對 MyBatis 所返回的對象作出更改,以防後患。

手動清空本地緩存:

void clearCache()

二級緩存

二級緩存是namespace級緩存,二級緩存會在同一 namespace中生效。

默認情況下,MyBatis 3 沒有開啓二級緩存,要開啓二級緩存,你需要在你的 SQL 映射文件(mapper.xml)中添加一行:

<cache/>

其實還需要在配置文件中把mybatis.configuration.cache-enabled設置爲true(默認爲true),若添加<cache/>標籤後緩存不生效,可以檢查是否將其設置爲了false

字面上看就是這樣。這個簡單語句的效果如下:

  • 映射語句文件中的所有 select 語句將會被緩存。
  • 映射語句文件中的所有 insert,update 和 delete 語句會刷新緩存。
  • 緩存會使用 Least Recently Used(LRU,最近最少使用的)算法來收回。
  • 根據時間表(比如 no Flush Interval,沒有刷新間隔), 緩存不會以任何時間順序 來刷新。
  • 緩存會存儲列表集合或對象(無論查詢方法返回什麼)的 1024 個引用。
  • 緩存會被視爲是 read/write(可讀/可寫)的緩存,意味着對象檢索不是共享的,而 且可以安全地被調用者修改,而不干擾其他調用者或線程所做的潛在修改。

所有的這些屬性都可以通過緩存元素的屬性來修改。比如:

<cache
  eviction="FIFO"
  flushInterval="60000"
  size="512"
  readOnly="true"/>

這個更高級的配置創建了一個 FIFO 緩存,並每隔 60 秒刷新,存數結果對象或列表的 512 個引用,而且返回的對象被認爲是隻讀的,因此在不同線程中的調用者之間修改它們會 導致衝突。

可用的收回策略有:

  • LRU – 最近最少使用的:移除最長時間不被使用的對象。
  • FIFO – 先進先出:按對象進入緩存的順序來移除它們。
  • SOFT – 軟引用:移除基於垃圾回收器狀態和軟引用規則的對象。
  • WEAK – 弱引用:更積極地移除基於垃圾收集器狀態和弱引用規則的對象。

默認的緩存回收策略是 LRU。

flushInterval(刷新間隔)可以被設置爲任意的正整數,而且它們代表一個合理的毫秒形式的時間段。默認情況是不設置,也就是沒有刷新間隔,緩存僅僅調用語句時刷新。

size(引用數目)可以被設置爲任意正整數,要記住你緩存的對象數目和你運行環境的 可用內存資源數目。默認值是 1024。

readOnly(只讀)屬性可以被設置爲 true 或 false。只讀的緩存會給所有調用者返回緩 存對象的相同實例。因此這些對象不能被修改。這提供了很重要的性能優勢。可讀寫的緩存 會返回緩存對象的拷貝(通過序列化) 。這會慢一些,但是安全,因此默認是 false。

若在SqlSession關閉時,SqlSession對應的本地緩存會自動轉化爲二級緩存。

自定義緩存

使用自定緩存,只需要實現MyBatis的Cache接口並在<cache/>中配置緩存類型:

<cache type="com.domain.something.MyCustomCache"/>

自定義緩存沒有使用過,如果大家有興趣可以參考MyBatis官方文檔自定義緩存部分

後記

這篇文章主要由MyBatis官方文檔整理而來,用於記錄我的學習過程,作爲2019年的開始,以後的學習都需要有產出物,否則學了之後很快就會忘記。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章