二級快取(Second-level)


Hibernate的 Session level 快取 隨著Session生命週期起始與消滅。

第一個 Hibernate 中的範例來說,在未使用二級快取的情況下,如果使用以下的程式片段來查詢資料:
Session session = sessionFactory.openSession();
User user1 = (User) session.load(User.class, new Integer(1));
user1.getName();
session.close();

session = sessionFactory.openSession();
User user2 = (User) session.load(User.class, new Integer(1));
user2.getName();
session.close();  

則Hibernate將會使用以下的SQL來進行資料查詢:
Hibernate: select user0_.id as id0_, user0_.name as name0_0_, user0_.age as age0_0_ from user user0_ where user0_.id=?
Hibernate: select user0_.id as id0_, user0_.name as name0_0_, user0_.age as age0_0_ from user user0_ where user0_.id=?

由於Session被關閉,Session level無法起作用,所以第二次的查詢仍必須向資料庫直接查詢。

Hibernate二級快取可以跨越數個Session,二級快取由同一個SessionFactory所建立的Session所共享,因而又稱為 SessionFactory level快取。

Hibernate本身並未提供二級快取的實現,而是藉由第三方(Third-party)產品來實現,Hibernate預設使用EHCache作為其 二級快取的實現,在最簡單的情況下,您只需在Hibernate下撰寫一個ehcache.xml作為EHCache的資源定義檔,可以在 Hibernate下載檔案中的etc目錄下找到一個已經撰寫好的ehcache.xml,以下撰寫最簡單的ehcache.xml:
  • ehcache.xml
<ehcache>
<diskStore path="java.io.tmpdir"/>
<defaultCache
maxElementsInMemory="10000"
eternal="false"
timeToIdleSeconds="120"
timeToLiveSeconds="120"
overflowToDisk="true"
/>
</ehcache>

將這個檔案放在Hibernate專案Classpath可存取到的路徑下, 接著重新運行上面的程式片段,您可以發現Hibernate將使用以下的SQL進行查詢:
Hibernate: select user0_.id as id0_, user0_.name as name0_0_, user0_.age as age0_0_ from user user0_ where user0_.id=?

二級快取被同一個SessionFactory所建立的Session實例所共享,所以即使關閉了Session,下一個Session仍可使用二級快 取,在查詢時,Session會先在Session level快取中查詢看有無資料,如果沒有就試著從二級快取中查詢資料,查到資料的話就直接返回該筆資料,所以在上例中,第二次無需再向資料庫進行SQL 查詢。

如果打算清除二級快取的資料,可以使用SessionFactory的evict()方法,例如:
sessionFactory.evict(User.class, user.getId());

如果打算在Hibernate中使用其它第三方產品進行快取,則可以在hibernate.cfg.xml中定義 hibernate.cache.provider_class屬性,例如:
  • hibernate.cfg.xml
<?xml version="1.0" encoding="utf-8"?> 
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">

<hibernate-configuration>

<session-factory>

....
<property name="hibernate.cache.provider_class">
org.hibernate.cache.HashtableCacheProvider
</property>
....

</session-factory>

</hibernate-configuration>

HashtableCache是Hibernate自己所提供的二級快取實現,不過性能與功能上有限,只用於開發時期的測試之用。

可以在映射文件中指定快取策略,使用<cache>標籤在映射實體或Collection上設定快取策略,例如:
  • User.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="onlyfun.caterpillar.User" table="user">
<cache usage="read-only"/>

<id name="id" column="id">
<generator class="native"/>
</id>

<property name="name" column="name"/>

<property name="age" column="age"/>

</class>

</hibernate-mapping>

可以設定的策略包括read-only、read-write、nonstrict-read-write與transactional,並不是每一個第 三方快取實現都支援所有的選項,每一個選項的使用時機與支援的產品,可以直接參考Hibernate官方參考手冊的 20.2. The Second Level Cache