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();
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=?
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。