国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

Hibernate二級緩存攻略

2019-11-18 14:50:15
字體:
來源:轉載
供稿:網友

  很多人對二級緩存都不太了解,或者是有錯誤的熟悉,我一直想寫一篇文章介紹一下hibernate的二級緩存的,今天終于忍不住了。

  我的經驗主要來自hibernate2.1版本,基本原理和3.0、3.1是一樣的,請原諒我的頑固不化。

  hibernate的session提供了一級緩存,每個session,對同一個id進行兩次load,不會發送兩條sql給數據庫,但是session關閉的時候,一級緩存就失效了。

  二級緩存是SessionFactory級別的全局緩存,它底下可以使用不同的緩存類庫,比如ehcache、oscache等,需要設置hibernate.cache.PRovider_class,我們這里用ehcache,在2.1中就是 hibernate.cache.provider_class=net.sf.hibernate.cache.EhCacheProvider假如使用查詢緩存,加上hibernate.cache.use_query_cache=true

  緩存可以簡單的看成一個Map,通過key在緩存里面找value。

  Class的緩存

  對于一條記錄,也就是一個PO來說,是根據ID來找的,緩存的key就是ID,value是POJO。無論list,load還是iterate,只要讀出一個對象,都會填充緩存。但是list不會使用緩存,而iterate會先取數據庫select id出來,然后一個id一個id的load,假如在緩存里面有,就從緩存取,沒有的話就去數據庫load。假設是讀寫緩存,需要設置:

<cache usage="read-write"/>
  假如你使用的二級緩存實現是ehcache的話,需要配置ehcache.xml

<cache name="com.xxx.pojo.Foo" maxElementsInMemory="500" eternal="false" timeToLiveSeconds="7200" timeToIdleSeconds="3600" overflowToDisk="true" />
  其中eternal表示緩存是不是永遠不超時,timeToLiveSeconds是緩存中每個元素(這里也就是一個POJO)的超時時間,假如eternal="false",超過指定的時間,這個元素就被移走了。timeToIdleSeconds是發呆時間,是可選的。當往緩存里面put的元素超過500個時,假如overflowToDisk="true",就會把緩存中的部分數據保存在硬盤上的臨時文件里面。

  每個需要緩存的class都要這樣配置。假如你沒有配置,hibernate會在啟動的時候警告你,然后使用defaultCache的配置,這樣多個class會共享一個配置。

  當某個ID通過hibernate修改時,hibernate會知道,于是移除緩存。

  這樣大家可能會想,同樣的查詢條件,第一次先list,第二次再iterate,就可以使用到緩存了。實際上這是很難的,因為你無法判定什么時候是第一次,而且每次查詢的條件通常是不一樣的,假如數據庫里面有100條記錄,id從1到100,第一次list的時候出了前50個id,第二次iterate的時候卻查詢到30至70號id,那么30-50是從緩存里面取的,51到70是從數據庫取的,共發送1+20條sql。所以我一直認為iterate沒有什么用,總是會有1+N的問題。

  (題外話:有說法說大型查詢用list會把整個結果集裝入內存,很慢,而iterate只select id比較好,但是大型查詢總是要分頁查的,誰也不會真的把整個結果集裝進來,假如一頁20條的話,iterate共需要執行21條語句,list雖然選擇若干字段,比iterate第一條select id語句慢一些,但只有一條語句,不裝入整個結果集hibernate還會根據數據庫方言做優化,比如使用MySQL的limit,整體看來應該還是list快。)

  假如想要對list或者iterate查詢的結果緩存,就要用到查詢緩存了

  查詢緩存

  首先需要配置hibernate.cache.use_query_cache=true

  假如用ehcache,配置ehcache.xml,注重hibernate3.0以后不是net.sf的包名了:

<cache name="net.sf.hibernate.cache.StandardQueryCache"
maxElementsInMemory="50" eternal="false" timeToIdleSeconds="3600"
timeToLiveSeconds="7200" overflowToDisk="true"/>
<cache name="net.sf.hibernate.cache.UpdateTimestampsCache"
maxElementsInMemory="5000" eternal="true" overflowToDisk="true"/>
  然后

query.setCacheable(true);//激活查詢緩存
query.setCacheRegion("myCacheRegion");//指定要使用的cacheRegion,可選
  第二行指定要使用的cacheRegion是myCacheRegion,即你可以給每個查詢緩存做一個單獨的配置,使用setCacheRegion來做這個指定,需要在ehcache.xml里面配置它:

<cache name="myCacheRegion" maxElementsInMemory="10" eternal="false" timeToIdleSeconds="3600" timeToLiveSeconds="7200" overflowToDisk="true" />
  假如省略第二行,不設置cacheRegion的話,那么會使用上面提到的標準查詢緩存的配置,也就是:net.sf.hibernate.cache.StandardQueryCache

  對于查詢緩存來說,緩存的key是根據hql生成的sql,再加上參數,分頁等信息(可以通過日志輸出看到,不過它的輸出不是很可讀,最好改一下它的代碼)。

  比如hql:



發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 丹江口市| 德化县| 宝坻区| 绍兴市| 龙南县| 黔江区| 宿迁市| 福贡县| 兴宁市| 铁力市| 玉树县| 拜泉县| 宜州市| 剑河县| 黎平县| 湘阴县| 永修县| 平潭县| 海城市| 镇安县| 洛南县| 阳曲县| 曲水县| 邮箱| 慈利县| 肃南| 偏关县| 南部县| 炉霍县| 游戏| 建德市| 柳河县| 太仆寺旗| 绍兴县| 陵水| 昌都县| 临湘市| 万源市| 专栏| 晋城| 会昌县|