Hibernate中二級(jí)緩存的配置和使用
(一)Hibernate的二級(jí)緩存策略的一般過(guò)程如下:
1) 條件查詢的時(shí)候,總是發(fā)出一條select * from table_name where …. (選擇所有字段)這樣的SQL語(yǔ)句查詢數(shù)據(jù)庫(kù),一次獲得所有的數(shù)據(jù)對(duì)象。
2) 把獲得的所有數(shù)據(jù)對(duì)象根據(jù)ID放入到第二級(jí)緩存中。
3) 當(dāng)Hibernate根據(jù)ID訪問(wèn)數(shù)據(jù)對(duì)象的時(shí)候,首先從Session一級(jí)緩存中查;查不到,如果配置了二級(jí)緩存,那么從二級(jí)緩存中查;查不到,再查詢數(shù)據(jù)庫(kù),把結(jié)果按照ID放入到緩存。
4) 刪除、更新、增加數(shù)據(jù)的時(shí)候,同時(shí)更新緩存。
Hibernate的二級(jí)緩存策略,是針對(duì)于ID查詢的緩存策略,對(duì)于條件查詢則毫無(wú)作用。為此,Hibernate提供了針對(duì)條件查詢的Query Cache。
(二)什么樣的數(shù)據(jù)適合存放到第二級(jí)緩存中?
1 很少被修改的數(shù)據(jù)
2 不是很重要的數(shù)據(jù),允許出現(xiàn)偶爾并發(fā)的數(shù)據(jù)
3 不會(huì)被并發(fā)訪問(wèn)的數(shù)據(jù)
4 參考數(shù)據(jù),指的是供應(yīng)用參考的常量數(shù)據(jù),它的實(shí)例數(shù)目有限,它的實(shí)例會(huì)被許多其他類的實(shí)例引用,實(shí)例極少或者從來(lái)不會(huì)被修改。
(三)不適合存放到第二級(jí)緩存的數(shù)據(jù)?
1 經(jīng)常被修改的數(shù)據(jù)
2 財(cái)務(wù)數(shù)據(jù),絕對(duì)不允許出現(xiàn)并發(fā)
3 與其他應(yīng)用共享的數(shù)據(jù)。
實(shí)踐部分:
使用EhCache配置二級(jí)緩存:
配置準(zhǔn)備:
1)把ehcache-1.2.3.jar加入到當(dāng)前應(yīng)用的classpath中。
2)在hibernate.cfg.xml文件中加入EhCache緩存插件的提供類。
- <!--配置緩存插件 -->
- <property name="hibernate.cache.provider_class">
- org.hibernate.cache.EhCacheProvider
- </property>
3)挎貝ehcache.xml文件到類路徑(項(xiàng)目工程的src目錄下),這個(gè)文件在Hibernate安裝目錄的etc下。
配置步驟:
Hibernate允許在類和集合的粒度上設(shè)置第二級(jí)緩存。在映射文件中,<class>和<set>元素都有一個(gè)<cache>子元素,這個(gè)子元素用來(lái)配置二級(jí)緩存。
示例:以category(產(chǎn)品類別)和product(產(chǎn)品)的映射為例:
1) 修改要配置緩存的那個(gè)持久化類的對(duì)象關(guān)系映射文件:
Category.hbm.xml
- <?xml version="1.0" encoding="utf-8"?>
- <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
- "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
- <hibernate-mapping>
- <class name="org.qiujy.domain.cachedemo.Category" table="categories">
- <!—
- 配置緩存,必須緊跟在class元素后面
- 對(duì)緩存中的Category對(duì)象采用讀寫(xiě)型的并發(fā)訪問(wèn)策略
- -->
- <cache usage="read-write"/>
- <id name="id" type="java.lang.Long">
- <column name="id" />
- <generator class="native" />
- </id>
- <!-- 配置版本號(hào),必須緊跟在id元素后面 -->
- <version name="version" column="version" type="java.lang.Long" />
- <property name="name" type="java.lang.String">
- <column name="name" length="32" not-null="true"/>
- </property>
- <property name="description" type="java.lang.String">
- <column name="description" length="255"/>
- </property>
- <set name="products" table="products" cascade="all" inverse="true">
- <!-- Hibernate只會(huì)緩存對(duì)象的簡(jiǎn)單屬性的值,
- 要緩存集合屬性,必須在集合元素中也加入<cache>子元素
- 而Hibernate僅僅是把與當(dāng)前持久對(duì)象關(guān)聯(lián)的對(duì)象的OID存放到緩存中。
- 如果希望把整個(gè)關(guān)聯(lián)的對(duì)象的所有數(shù)據(jù)都存入緩存,
- 則要在相應(yīng)關(guān)聯(lián)的對(duì)象的映射文件中配置<cache>元素
- -->
- <cache usage="read-write"/>
- <key column="categoryId" not-null="true"/>
- <one-to-many class="org.qiujy.domain.cachedemo.Product"/>
- </set>
- </class>
- </hibernate-mapping>
Product.hbm.xml
- <?xml version="1.0" encoding="utf-8"?>
- <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
- "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
- <hibernate-mapping>
- <class name="org.qiujy.domain.cachedemo.Product" table="products">
- <cache usage="read-write"/>
- <id name="id" type="java.lang.Long">
- <column name="id" />
- <generator class="native" />
- </id>
- <!-- 配置版本號(hào),必須緊跟在id元素后面 -->
- <version name="version" column="version" type="java.lang.Long" />
- <property name="name" type="java.lang.String">
- <column name="name" length="32" not-null="true"/>
- </property>
- <property name="description" type="java.lang.String">
- <column name="description" length="255"/>
- </property>
- <property name="unitCost" type="java.lang.Double">
- <column name="unitCost" />
- </property>
- <property name="pubTime" type="java.util.Date">
- <column name="pubTime" not-null="true" />
- </property>
- <many-to-one name="category"
- column="categoryId"
- class="org.qiujy.domain.cachedemo.Category"
- cascade="save-update"
- not-null="true">
- </many-to-one>
- </class>
- </hibernate-mapping>
2)編輯ehcache.xml文件:
- <ehcache>
- <diskStore path="c:\\ehcache\"/>
- <defaultCache
- maxElementsInMemory="10000"
- eternal="false"
- timeToIdleSeconds="120"
- timeToLiveSeconds="120"
- overflowToDisk="true"
- />
- <!-- 設(shè)置Category類的緩存的數(shù)據(jù)過(guò)期策略 -->
- <cache name="org.qiujy.domain.cachedemo.Category"
- maxElementsInMemory="100"
- eternal="true"
- timeToIdleSeconds="0"
- timeToLiveSeconds="0"
- overflowToDisk="false"
- />
- <!-- 設(shè)置Category類的products集合的緩存的數(shù)據(jù)過(guò)期策略 -->
- <cache name="org.qiujy.domain.cachedemo.Category.products"
- maxElementsInMemory="500"
- eternal="false"
- timeToIdleSeconds="300"
- timeToLiveSeconds="600"
- overflowToDisk="true"
- />
- <cache name="org.qiujy.domain.cachedemo.Product"
- maxElementsInMemory="500"
- eternal="false"
- timeToIdleSeconds="300"
- timeToLiveSeconds="600"
- overflowToDisk="true"
- />
- </ehcache>
在Spring托管的Hibernate中使用二級(jí)緩存 1.在spring的配置文件中,hibernate部分加入 xml 代碼
【編輯推薦】