Hibernate二級緩存 ehcache
二級緩存稱為進程級緩存或SessionFactory級緩存,它可以被所有session共享,它的生命周期伴隨著SessionFactory的生命周期存在和消亡。
第一步:復制ehcache.xml文件到src目錄下,配置二級緩存
<defaultCache maxElementsInMemory="100" --設置緩存中的最大對象數eternal="false" --設置緩存中的對象是不是永不過期,如果設置為true,timeToIdleSeconds和timeToLiveSeconds就沒有意義了
timeToIdleSeconds="120" --設置空閑時間,多少時間之內沒有訪問自動清除
timeToLiveSeconds="120" --對象存活時間,一般情況下大于等于timeToIdleSeconds
diskPersistent="true"
overflowToDisk="true" --將超出的對象存到硬盤上,存儲位置設置,例如:<diskStore path="E:/cache"/> /> </pre>
也可以配置某個特定對象的,如(但一般情況是使用缺省的就可以了):<cache name="uuser" maxElementsInMemory="10000" eternal="false" timeToIdleSeconds="300" timeToLiveSeconds="600" overflowToDisk="true" />
第二步:啟用二級緩存
1.啟用二級緩存,默認是啟用的:
<property name="cache.use_second_level_cache">true</property>
2.設置緩存產品,如:
<property name="cache.provider_class">org.hibernate.cache.EhCacheProvider</property>
3.指定哪些類使用二級緩存(兩種方法):
a.在hibernate.cfg.xml配置文件中定義
如:<class-cache class="lovo.po.User" usage="read-write"/>
b.在映射文件中定義
如:在<class>節點下:<cache usage="read-only"/>
有些類設置二級緩存,有些類不設置二級緩存;所以如果在映射文件中定義二級緩存的話會導致<cache>標簽遍布在文件中,
因此常用第一種,即在hibernate.cfg.xml配置文件中定義,這樣便于維護和管理。
第三步:二級緩存實例
1.打開兩次session,調用load查詢
代碼部分:
Session session = this.sf.openSession(); User u1 = (User) session.load(User.class, 1); System.out.println("用戶姓名:"+u1.getName()); session.close(); //關閉seeion,釋放一級緩存 session = this.sf.openSession(); //重新開啟session User u = (User) session.load(User.class, 1); System.out.println("用戶姓名:"+u.getName()); session.close();
執行部分:
如果設置<property name="cache.use_second_levelcache">false</property>,執行結果如下:
Hibernate: select
user0.id as id00, user0_.name as name00, user0_.password as password00, user0_.createTime as createTime00,
user0_.expireTime as expireTime00 from tuser user0 where user0.id=?
用戶姓名:wangwu
Hibernate:select user0.id as id00,user0_.name as name00,user0_.password as password00,
user0_.createTime as createTime00,user0_.expireTime as expireTime00 from
tuser user0 where user0_.id=?
用戶姓名:wangwu
如果設置<property name="cache.use_second_levelcache">true</property>,執行結果如下:
Hibernate:
select user0.id as id00, user0_.name as name00, user0_.password as password00,
user0_.createTime as createTime00, user0_.expireTime as expireTime00 from tuser user0
where user0_.id=?
用戶姓名:wangwu
用戶姓名:wangwu
可見,二級緩存是起作用了的。用get測試是同樣的結果
2.在1的代碼中加入this.sf.evict(User.class),清除二級緩存中的對象,執行結果就跟設置了<property name="cache.use_second_level_cache">false</property>的效果相同。
當然也可以指定清除哪些對象:this.sf.evict(User.class,1),結果很簡單。
3.一級緩存和二級緩存的交互(CacheMode:CacheMode.NORMAL、CacheMode.PUT、CacheMode.GET)
代碼部分:
Session session = this.sf.openSession(); //CacheMode.GET只向二級緩存中讀數據不寫 session.setCacheMode(CacheMode.GET); ……1 User u1 = (User) session.load(User.class, 1); ……2 //代碼1和代碼2順序不能顛倒,否則CacheMode.GET對1號用戶不起作用 System.out.println("用戶姓名:"+u1.getName()); session.close();session = this.sf.openSession(); //重新開啟session User u = (User) session.load(User.class, 1); System.out.println("用戶姓名:"+u.getName()); session.close(); </pre>
執行結果:
Hibernate: select user0_.id as id00,
user0_.name as name00, user0_.password as password00, user0_.createTime as createTime00, user0_.expireTime as expireTime00
from tuser user0 where user0.id=?
用戶姓名:wangwu
Hibernate:
select
user0.id as id00, user0_.name as name00, user0_.password as password00, user0_.createTime as createTime00,
user0_.expireTime as expireTime00 from tuser user0 where
user0_.id=?
用戶姓名:wangwu
默認情況下又讀又寫,即為CacheMode.NORMAL
CacheMode.PUT只寫不讀