MyBatis之查询缓存
MyBatis之查詢緩存
2017/09/30正如大多數(shù)持久層框架一樣,MyBatis同樣也提供了對(duì)查詢數(shù)據(jù)的緩存支持。今后我們要學(xué)習(xí)的SpringMVC框架屬于系統(tǒng)控制層,它也有它的緩存區(qū)域,對(duì)響應(yīng)的jsp頁(yè)面進(jìn)行緩存;Spring屬于系統(tǒng)業(yè)務(wù)層,它的緩存區(qū)域?qū)I(yè)務(wù)數(shù)據(jù)進(jìn)行緩存。而它們?nèi)叩木彺鎱^(qū)域都存在于用戶要訪問(wèn)數(shù)據(jù)庫(kù)的中間,當(dāng)用戶向數(shù)據(jù)庫(kù)中發(fā)送查詢請(qǐng)求時(shí),系統(tǒng)會(huì)先在它們?nèi)叩木彺鎱^(qū)域看是否有滿足用戶要求的查詢數(shù)據(jù),若有則直接從緩存區(qū)域返回信息;若沒(méi)有則從數(shù)據(jù)庫(kù)中進(jìn)行查詢。
1.緩存的意義
將用戶經(jīng)常查詢的數(shù)據(jù)放在緩存(內(nèi)存)中,用戶去查詢數(shù)據(jù)就不用從磁盤(pán)上(關(guān)系型數(shù)據(jù)庫(kù)數(shù)據(jù)文件)查詢,從緩存中查詢,從而提高查詢效率,解決了高并發(fā)系統(tǒng)的性能問(wèn)題。
2.MyBatis持久層緩存
MyBatis提供了一級(jí)緩存和二級(jí)緩存,概念圖如下:
如圖中所示,MyBatis一級(jí)緩存是一個(gè)SqlSession級(jí)別,SqlSession只能訪問(wèn)自己的一級(jí)緩存的數(shù)據(jù),二級(jí)緩存是跨SqlSession,是mapper級(jí)別的緩存,對(duì)于mapper級(jí)別的緩存不同的Sqlsession是可以共享的。
3.MyBatis一級(jí)緩存
3.1原理
對(duì)于圖的解釋:第一次發(fā)出一個(gè)查詢sql,sql查詢結(jié)果寫(xiě)入sqlsession的一級(jí)緩存中,緩存使用的數(shù)據(jù)結(jié)構(gòu)是一個(gè)map<key,value>。
key:hashcode+sql+sql輸入?yún)?shù)+輸出參數(shù)(sql的唯一標(biāo)識(shí))
value:用戶信息
同一個(gè)sqlsession再次發(fā)出相同的sql,就從緩存中取,而不走數(shù)據(jù)庫(kù)。如果兩次中間出現(xiàn)commit操作(修改、添加、刪除),本sqlsession中的一級(jí)緩存區(qū)域全部清空,下次再去緩存中查詢不到所以要從數(shù)據(jù)庫(kù)查詢,從數(shù)據(jù)庫(kù)查詢到再寫(xiě)入緩存。即對(duì)于查詢操作,每次查詢都先從緩存中查詢,如果緩存中查詢到數(shù)據(jù)則將緩存數(shù)據(jù)直接返回,如果緩存中查詢不到就從數(shù)據(jù)庫(kù)查詢。
3.2一級(jí)緩存配置
mybatis默認(rèn)支持一級(jí)緩存不需要配置。
注意:mybatis和spring整合后進(jìn)行mapper代理開(kāi)發(fā),不支持一級(jí)緩存,mybatis和spring整合,spring按照mapper的模板去生成mapper代理對(duì)象,模板中在最后統(tǒng)一關(guān)閉sqlsession。
3.3一級(jí)緩存的測(cè)試
測(cè)試代碼:
運(yùn)行,在控制臺(tái)看到輸出:
發(fā)現(xiàn)打印兩次結(jié)果只有第一次向數(shù)據(jù)庫(kù)中發(fā)送了查詢請(qǐng)求,第二次是直接從緩存中返回的數(shù)據(jù)。接下來(lái)我們?cè)谥虚g加入更新數(shù)據(jù)的操作:
然后在控制臺(tái)中看到打印過(guò)程:
可以發(fā)現(xiàn)向數(shù)據(jù)庫(kù)發(fā)送了三次請(qǐng)求。
4.MyBatis二級(jí)緩存
4.1二級(jí)緩存原理
二級(jí)緩存的范圍是mapper級(jí)別(mapper同一個(gè)命名空間),mapper以命名空間為單位創(chuàng)建緩存數(shù)據(jù)結(jié)構(gòu),結(jié)構(gòu)是map<key、value>。
過(guò)程:每次查詢先看是否開(kāi)啟二級(jí)緩存,如果開(kāi)啟從二級(jí)緩存的數(shù)據(jù)結(jié)構(gòu)中取緩存數(shù)據(jù),如果從二級(jí)緩存沒(méi)有取到,再?gòu)囊患?jí)緩存中找,如果一級(jí)緩存也沒(méi)有,從數(shù)據(jù)庫(kù)查詢。
4.2二級(jí)緩存的配置
不像一級(jí)緩存那樣mybatis自動(dòng)開(kāi)啟一級(jí)緩存,mybatis是默認(rèn)關(guān)閉二級(jí)緩存的,所以我們需要需要進(jìn)行兩個(gè)操作才能開(kāi)啟二級(jí)緩存:
1.在核心配置文件SqlMapperConfig.xml中加入
屬性值cacheEnabled表示對(duì)在此配置文件下的所有cache 進(jìn)行全局性開(kāi)/關(guān)設(shè)置,它的可選值為true|false,默認(rèn)值為true.
2.在你的Mapper映射文件中添加一行:<cache />?,表示此mapper開(kāi)啟二級(jí)緩存。
4.3查詢結(jié)果映射的pojo序列化
mybatis二級(jí)緩存需要將查詢結(jié)果映射的pojo實(shí)現(xiàn) java.io.serializable接口,如果不實(shí)現(xiàn)則拋出異常:
org.apache.ibatis.cache.CacheException: Error serializing object. Cause: java.io.NotSerializableException: cn.itcast.mybatis.po.User
二級(jí)緩存可以將內(nèi)存的數(shù)據(jù)寫(xiě)到磁盤(pán),存在對(duì)象的序列化和反序列化,所以要實(shí)現(xiàn)java.io.serializable接口。
如果結(jié)果映射的pojo中還包括了pojo,都要實(shí)現(xiàn)java.io.serializable接口。
4.4二級(jí)緩存的禁用
對(duì)于變化頻率較高的sql,需要禁用二級(jí)緩存:在statement中設(shè)置useCache=false可以禁用當(dāng)前select語(yǔ)句的二級(jí)緩存,即每次查詢都會(huì)發(fā)出sql去查詢,默認(rèn)情況是true,即該sql使用二級(jí)緩存。
<select id="findOrderListResultMap" resultMap="ordersUserMap" useCache="false">
4.5刷新緩存
將二級(jí)緩存進(jìn)行刷新操作有兩種方式:
- 如果sqlsession操作commit操作,對(duì)二級(jí)緩存進(jìn)行刷新(全局清空)。
- 設(shè)置statement的flushCache是否刷新緩存,默認(rèn)值是true。
4.6測(cè)試代碼
分別將中間更新用戶信息的代碼注釋:發(fā)現(xiàn)兩次打印用戶信息的操作只向數(shù)據(jù)庫(kù)中發(fā)送一次查詢請(qǐng)求;去掉注釋:三次操作向數(shù)據(jù)庫(kù)中發(fā)送三次請(qǐng)求。
4.7mybatis的cache參數(shù)(了解)
mybatis的cache參數(shù)只適用于mybatis維護(hù)緩存。
- flushInterval(刷新間隔):可以被設(shè)置為任意的正整數(shù),而且它們代表一個(gè)合理的毫秒形式的時(shí)間段。默認(rèn)情況是不設(shè)置,也就是沒(méi)有刷新間隔,緩存僅僅調(diào)用語(yǔ)句時(shí)刷新。
- size(引用數(shù)目):可以被設(shè)置為任意正整數(shù),要記住你緩存的對(duì)象數(shù)目和你運(yùn)行環(huán)境的可用內(nèi)存資源數(shù)目。默認(rèn)值是1024。
- readOnly(只讀):屬性可以被設(shè)置為true或false。只讀的緩存會(huì)給所有調(diào)用者返回緩存對(duì)象的相同實(shí)例。因此這些對(duì)象不能被修改。這提供了很重要的性能優(yōu)勢(shì)。可讀寫(xiě)的緩存會(huì)返回緩存對(duì)象的拷貝(通過(guò)序列化)。這會(huì)慢一些,但是安全,因此默認(rèn)是false。
如下例子:
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
這個(gè)更高級(jí)的配置創(chuàng)建了一個(gè) FIFO 緩存,并每隔 60 秒刷新,存數(shù)結(jié)果對(duì)象或列表的 512 個(gè)引用,而且返回的對(duì)象被認(rèn)為是只讀的,因此在不同線程中的調(diào)用者之間修改它們會(huì)導(dǎo)致沖突??捎玫氖栈夭呗杂? 默認(rèn)的是 LRU:
- LRU – 最近最少使用的:移除最長(zhǎng)時(shí)間不被使用的對(duì)象。
- FIFO – 先進(jìn)先出:按對(duì)象進(jìn)入緩存的順序來(lái)移除它們。
- SOFT – 軟引用:移除基于垃圾回收器狀態(tài)和軟引用規(guī)則的對(duì)象。
- WEAK – 弱引用:更積極地移除基于垃圾收集器狀態(tài)和弱引用規(guī)則的對(duì)象。
4.8二級(jí)緩存的應(yīng)用場(chǎng)景
對(duì)查詢頻率高,變化頻率低的數(shù)據(jù)建議使用二級(jí)緩存。
對(duì)于訪問(wèn)多的查詢請(qǐng)求且用戶對(duì)查詢結(jié)果實(shí)時(shí)性要求不高,此時(shí)可采用mybatis二級(jí)緩存技術(shù)降低數(shù)據(jù)庫(kù)訪問(wèn)量,提高訪問(wèn)速度,業(yè)務(wù)場(chǎng)景比如:耗時(shí)較高的統(tǒng)計(jì)分析sql、電話賬單查詢sql等。
實(shí)現(xiàn)方法如下:通過(guò)設(shè)置刷新間隔時(shí)間,由mybatis每隔一段時(shí)間自動(dòng)清空緩存,根據(jù)數(shù)據(jù)變化頻率設(shè)置緩存刷新間隔flushInterval,比如設(shè)置為30分鐘、60分鐘、24小時(shí)等,根據(jù)需求而定。
4.9二級(jí)緩存的局限性
mybatis二級(jí)緩存對(duì)細(xì)粒度的數(shù)據(jù)級(jí)別的緩存實(shí)現(xiàn)不好,比如如下需求:對(duì)商品信息進(jìn)行緩存,由于商品信息查詢?cè)L問(wèn)量大,但是要求用戶每次都能查詢最新的商品信息,此時(shí)如果使用mybatis的二級(jí)緩存就無(wú)法實(shí)現(xiàn)當(dāng)一個(gè)商品變化時(shí)只刷新該商品的緩存信息而不刷新其它商品的信息,因?yàn)閙ybaits的二級(jí)緩存區(qū)域以mapper為單位劃分,當(dāng)一個(gè)商品信息變化會(huì)將所有商品信息的緩存數(shù)據(jù)全部清空。解決此類問(wèn)題需要在業(yè)務(wù)層根據(jù)需求對(duì)數(shù)據(jù)有針對(duì)性緩存。
5.MyBatis和ehcache緩存框架整合
5.1分布緩存
將緩存數(shù)據(jù)進(jìn)行分布式管理,這個(gè)概念我們以后會(huì)學(xué)
5.3MyBatis和ehcache整合思路
通過(guò)MyBatis和ehcache框架進(jìn)行整合,就可以把緩存數(shù)據(jù)的管理托管給ehcache。
首先我們看看MyBatis自己的二級(jí)緩存,它在自己內(nèi)部提供了一個(gè)cache接口,我們只要實(shí)現(xiàn)了cache接口就可以把緩存數(shù)據(jù)靈活的管理起來(lái)。
要將MyBatis和ehcache整合,我們首先需要下載ehcache的jar包:
- ehcache-core.jar
- mybatis-ehcache.jar
然后需要添加一個(gè)ehcache的配置文件ehcache.xml:
然后在mapper.xml中添加ehcache的配置:
然后便可以進(jìn)行測(cè)試。
學(xué)完以上7篇文章,你就可以使用MyBatis去搭建一個(gè)項(xiàng)目了.
總結(jié)
以上是生活随笔為你收集整理的MyBatis之查询缓存的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: SteamVR导致场景相机不正常
- 下一篇: Unity打包失败解决方案