欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

mybatis教程之查詢緩存(一級(jí)緩存二級(jí)緩存和整合ehcache)

 更新時(shí)間:2017年05月03日 09:38:20   作者:SIHAIloveYAN  
這篇文章主要介紹了mybatis教程之查詢緩存(一級(jí)緩存二級(jí)緩存和整合ehcache),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

1 緩存的意義

將用戶經(jīng)常查詢的數(shù)據(jù)放在緩存(內(nèi)存)中,用戶去查詢數(shù)據(jù)就不用從磁盤上(關(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 一級(jí)緩存

3.1 原理

第一次發(fā)出一個(gè)查詢sql,sql查詢結(jié)果寫入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ù)查詢到再寫入緩存。

每次查詢都先從緩存中查詢:


 

如果緩存中查詢到則將緩存數(shù)據(jù)直接返回。

如果緩存中查詢不到就從數(shù)據(jù)庫(kù)查詢:

3.2 一級(jí)緩存配置

mybatis默認(rèn)支持一級(jí)緩存不需要配置。

注意:mybatis和spring整合后進(jìn)行mapper代理開發(fā),不支持一級(jí)緩存,mybatis和spring整合,spring按照mapper的模板去生成mapper代理對(duì)象,模板中在最后統(tǒng)一關(guān)閉sqlsession。

3.3 一級(jí)緩存測(cè)試

//一級(jí)緩存 
  @Test 
  public void testCache1() throws Exception { 
 
    SqlSession sqlSession = sqlSessionFactory.openSession(); 
    UserMapper userMapper = sqlSession.getMapper(UserMapper.class); 
     
    //第一次查詢用戶id為1的用戶 
    User user = userMapper.findUserById(1); 
    System.out.println(user); 
     
    //中間修改用戶要清空緩存,目的防止查詢出臟數(shù)據(jù) 
    /*user.setUsername("測(cè)試用戶2"); 
    userMapper.updateUser(user); 
    sqlSession.commit();*/ 
     
    //第二次查詢用戶id為1的用戶 
    User user2 = userMapper.findUserById(1); 
    System.out.println(user2); 
     
    sqlSession.close(); 
     
 
  } 

4 二級(jí)緩存

4.1 原理

二級(jí)緩存的范圍是mapper級(jí)別(mapper同一個(gè)命名空間),mapper以命名空間為單位創(chuàng)建緩存數(shù)據(jù)結(jié)構(gòu),結(jié)構(gòu)是map<key、value>。

每次查詢先看是否開啟二級(jí)緩存,如果開啟從二級(jí)緩存的數(shù)據(jù)結(jié)構(gòu)中取緩存數(shù)據(jù),

 

如果從二級(jí)緩存沒有取到,再?gòu)囊患?jí)緩存中找,如果一級(jí)緩存也沒有,從數(shù)據(jù)庫(kù)查詢。

4.2 mybatis二級(jí)緩存配置

在核心配置文件SqlMapConfig.xml中加入

<setting name="cacheEnabled" value="true"/> 

要在你的Mapper映射文件中添加一行:  <cache /> ,表示此mapper開啟二級(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: com.sihai.mybatis.po.User

二級(jí)緩存可以將內(nèi)存的數(shù)據(jù)寫到磁盤,存在對(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 刷新緩存

如果sqlsession操作commit操作,對(duì)二級(jí)緩存進(jìn)行刷新(全局清空)。

設(shè)置statement的flushCache是否刷新緩存,默認(rèn)值是true。

4.6 測(cè)試代碼

//二級(jí)緩存的測(cè)試 
  @Test 
  public void testCache2() throws Exception { 
 
    SqlSession sqlSession1 = sqlSessionFactory.openSession(); 
    SqlSession sqlSession2 = sqlSessionFactory.openSession(); 
    SqlSession sqlSession3 = sqlSessionFactory.openSession(); 
    UserMapper userMapper1 = sqlSession1.getMapper(UserMapper.class); 
    UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class); 
    UserMapper userMapper3 = sqlSession3.getMapper(UserMapper.class); 
     
    //第一次查詢用戶id為1的用戶 
    User user = userMapper1.findUserById(1); 
    System.out.println(user); 
    sqlSession1.close(); 
     
    //中間修改用戶要清空緩存,目的防止查詢出臟數(shù)據(jù) 
    /*user.setUsername("測(cè)試用戶2"); 
    userMapper3.updateUser(user); 
    sqlSession3.commit(); 
    sqlSession3.close();*/ 
     
     
    //第二次查詢用戶id為1的用戶 
    User user2 = userMapper2.findUserById(1); 
    System.out.println(user2); 
     
    sqlSession2.close(); 
     
 
  } 

4.7 mybatis的cache參數(shù)

mybatis的cache參數(shù)只適用于mybatis維護(hù)緩存。

flushInterval(刷新間隔)可以被設(shè)置為任意的正整數(shù),而且它們代表一個(gè)合理的毫秒形式的時(shí)間段。默認(rèn)情況是不設(shè)置,也就是沒有刷新間隔,緩存僅僅調(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ì)。可讀寫的緩存會(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:

1. LRU – 最近最少使用的:移除最長(zhǎng)時(shí)間不被使用的對(duì)象。

2. FIFO – 先進(jìn)先出:按對(duì)象進(jìn)入緩存的順序來(lái)移除它們。

3. SOFT – 軟引用:移除基于垃圾回收器狀態(tài)和軟引用規(guī)則的對(duì)象。

4. WEAK – 弱引用:更積極地移除基于垃圾收集器狀態(tài)和弱引用規(guī)則的對(duì)象。

5 mybatis和ehcache緩存框架整合

mybatis二級(jí)緩存通過(guò)ehcache維護(hù)緩存數(shù)據(jù)。

5.1 分布緩存

將緩存數(shù)據(jù)數(shù)據(jù)進(jìn)行分布式管理。

5.2 mybatis和ehcache思路

通過(guò)mybatis和ehcache框架進(jìn)行整合,就可以把緩存數(shù)據(jù)的管理托管給ehcache。

在mybatis中提供一個(gè)cache接口,只要實(shí)現(xiàn)cache接口就可以把緩存數(shù)據(jù)靈活的管理起來(lái)。

mybatis中默認(rèn)實(shí)現(xiàn):

5.3 下載和ehcache整合的jar包

 

ehcache對(duì)cache接口的實(shí)現(xiàn)類:

5.4 配置ehcache.xml

<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  xsi:noNamespaceSchemaLocation="../config/ehcache.xsd"> 
  <!--diskStore:緩存數(shù)據(jù)持久化的目錄 地址 --> 
  <diskStore path="F:\develop\ehcache" /> 
  <defaultCache  
    maxElementsInMemory="1000"  
    maxElementsOnDisk="10000000" 
    eternal="false"  
    overflowToDisk="false"  
    diskPersistent="true" 
    timeToIdleSeconds="120" 
    timeToLiveSeconds="120"  
    diskExpiryThreadIntervalSeconds="120" 
    memoryStoreEvictionPolicy="LRU"> 
  </defaultCache> 
</ehcache> 

5.5 整合測(cè)試

在mapper.xml添加ehcache配置:

<!-- 開啟二級(jí)緩存 --> 
  <!-- 單位:毫秒 --> 
  <cache type="org.mybatis.caches.ehcache.EhcacheCache"> 
    <property name="timeToIdleSeconds" value="12000"/> 
    <property name="timeToLiveSeconds" value="3600"/> 
    <!-- 同ehcache參數(shù)maxElementsInMemory --> 
    <property name="maxEntriesLocalHeap" value="1000"/> 
    <!-- 同ehcache參數(shù)maxElementsOnDisk --> 
    <property name="maxEntriesLocalDisk" value="10000000"/> 
    <property name="memoryStoreEvictionPolicy" value="LRU"/> 
  </cache> 

6 二級(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ù)需求而定。

7 mybatis局限性

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ì)性緩存。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • java?WebSocket?服務(wù)端實(shí)現(xiàn)代碼

    java?WebSocket?服務(wù)端實(shí)現(xiàn)代碼

    WebSocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò)協(xié)議。它實(shí)現(xiàn)了瀏覽器與服務(wù)器全雙工(full-duplex)通信——允許服務(wù)器主動(dòng)發(fā)送信息給客戶端,這篇文章主要介紹了java?WebSocket?服務(wù)端代碼,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-02-02
  • Spring Session實(shí)現(xiàn)分布式session的簡(jiǎn)單示例

    Spring Session實(shí)現(xiàn)分布式session的簡(jiǎn)單示例

    本篇文章主要介紹了Spring Session實(shí)現(xiàn)分布式session的簡(jiǎn)單示例,具有很好的參考價(jià)值。下面跟著小編一起來(lái)看下吧
    2017-05-05
  • jackson 實(shí)體轉(zhuǎn)json 為NULL或者為空不參加序列化(實(shí)例講解)

    jackson 實(shí)體轉(zhuǎn)json 為NULL或者為空不參加序列化(實(shí)例講解)

    下面小編就為大家?guī)?lái)一篇jackson 實(shí)體轉(zhuǎn)json 為NULL或者為空不參加序列化(實(shí)例講解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-10-10
  • Spring Cloud超詳細(xì)i講解Feign自定義配置與使用

    Spring Cloud超詳細(xì)i講解Feign自定義配置與使用

    這篇文章主要介紹了SpringCloud Feign自定義配置與使用,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-06-06
  • Springboot事務(wù)失效的幾種情況解讀

    Springboot事務(wù)失效的幾種情況解讀

    這篇文章主要介紹了Springboot事務(wù)失效的幾種情況解讀,因?yàn)镾pring AOP默認(rèn)使用動(dòng)態(tài)代理,會(huì)給被代理的類生成一個(gè)代理類,事務(wù)相關(guān)的操作都通過(guò)代理來(lái)完成,使用內(nèi)部方法調(diào)用時(shí),使用的是實(shí)例調(diào)用,沒有通過(guò)代理類調(diào)用方法,因此事務(wù)不會(huì)檢測(cè)到失敗,需要的朋友可以參考下
    2023-10-10
  • 使用AbstractRoutingDataSource實(shí)現(xiàn)數(shù)據(jù)源動(dòng)態(tài)切換的實(shí)例

    使用AbstractRoutingDataSource實(shí)現(xiàn)數(shù)據(jù)源動(dòng)態(tài)切換的實(shí)例

    AbstractRoutingDataSource 是 Spring 框架提供的一個(gè)抽象類,用于實(shí)現(xiàn)動(dòng)態(tài)數(shù)據(jù)源路由,這個(gè)類主要用于多數(shù)據(jù)源場(chǎng)景,其中可以根據(jù)不同的條件動(dòng)態(tài)地切換到不同的數(shù)據(jù)源,本文給大家介紹了如何使用AbstractRoutingDataSource實(shí)現(xiàn)數(shù)據(jù)源動(dòng)態(tài)切換,需要的朋友可以參考下
    2024-03-03
  • Jpa數(shù)據(jù)操作以及@Query和@Modifying注解使用方式

    Jpa數(shù)據(jù)操作以及@Query和@Modifying注解使用方式

    這篇文章主要介紹了Jpa數(shù)據(jù)操作以及@Query和@Modifying注解使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • Java編程異常簡(jiǎn)單代碼示例

    Java編程異常簡(jiǎn)單代碼示例

    這篇文章主要介紹了Java編程異常簡(jiǎn)單代碼示例,還是比較不錯(cuò)的,這里分享個(gè)大家,需要的朋友可以參考下。
    2017-11-11
  • 解決IDEA2021版compiler.automake.allow.when.app.running不存在的問(wèn)題

    解決IDEA2021版compiler.automake.allow.when.app.running不存在的問(wèn)題

    很多文章介紹IntelliJ IDEA開啟熱部署功能都會(huì)寫到在IntelliJ IDEA中的注冊(cè)表中開啟compiler.automake.allow.when.app.running選項(xiàng),此選項(xiàng)在IntelliJ IDEA 2021.2之后的版本遷移到高級(jí)設(shè)置中,下面看下設(shè)置方法
    2021-09-09
  • springboot集成es插入和查詢的簡(jiǎn)單使用示例詳解

    springboot集成es插入和查詢的簡(jiǎn)單使用示例詳解

    這篇文章主要介紹了springboot集成es 插入和查詢的簡(jiǎn)單使用,本文分步驟結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-08-08

最新評(píng)論