|
第一级缓存
|
第二级缓存
|
存放数据的形式
|
相互关联的持久化对象
|
对象的散装数据
|
缓存的范围
|
事务范围,每个事务都有单独的第一级缓存
|
进程范围或集群范围,缓存被同一个进程或集群范围内的所有事务共享
|
并发访问策略
|
由于每个事务都拥有单独的第一级缓存,不会出现并发问题,无需提供并发访问策略
|
由于多个事务会同时访问第二级缓存中相同数据,因此必须提供适当的并发访问策略,来保证特定的事务隔离级别
|
数据过期策略
|
没有提供数据过期策略。处于一级缓存中的对象永远不会过期,除非应用程序显式清空缓存或者清除特定的对象
|
必须提供数据过期策略,如基于内存的缓存中的对象的最大数目,允许对象处于缓存中的最长时间,以及允许对象处于缓存中的最长空闲时间
|
物理存储介质
|
内存
|
内存和硬盘。对象的散装数据首先存放在基于内在的缓存中,当内存中对象的数目达到数据过期策略中指定上限时,就会把其余的对象写入基于硬盘的缓存中。
|
缓存的软件实现
|
在Hibernate的Session的实现中包含了缓存的实现
|
由第三方提供,Hibernate仅提供了缓存适配器(CacheProvider)。用于把特定的缓存插件集成到Hibernate中。
|
启用缓存的方式
|
只要应用程序通过Session接口来执行保存、更新、删除、加载和查询数据库数据的操作,Hibernate就会启用第一级缓存,把数据库中的数据以对象的形式拷贝到缓存中,对于批量更新和批量删除操作,如果不希望启用第一级缓存,可以绕过Hibernate API,直接通过JDBC API来执行指操作。
|
用户可以在单个类或类的单个集合的粒度上配置第二级缓存。如果类的实例被经常读但很少被修改,就可以考虑使用第二级缓存。只有为某个类或集合配置了第二级缓存,Hibernate在运行时才会把它的实例加入到第二级缓存中。
|
用户管理缓存的方式
|
第一级缓存的物理介质为内存,由于内存容量有限,必须通过恰当的检索策略和检索方式来限制加载对象的数目。Session的evit()方法可以显式清空缓存中特定对象,但这种方法不值得推荐。
|
第二级缓存的物理介质可以是内存和硬盘,因此第二级缓存可以存放大量的数据,数据过期策略的maxElementsInMemory属性值可以控制内存中的对象数目。管理第二级缓存主要包括两个方面:选择需要使用第二级缓存的持久类,设置合适的并发访问策略:选择缓存适配器,设置合适的数据过期策略。
|
<!--
配置缓存插件
-->
<
property
name
=
"hibernate.cache.provider_class"
>
org.hibernate.cache.EhCacheProvider
</
property
>
|
<?
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
元素后面
对缓存中的
Category
对象采用读写型的并发访问策略
-->
<
cache
usage
=
"read-write"
/>
<
id
name
=
"id"
type
=
"java.lang.Long"
>
<
column
name
=
"id"
/>
<
generator
class
=
"native"
/>
</
id
>
<!--
配置版本号
,
必须紧跟在
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
只会缓存对象的简单属性的值
,
要缓存集合属性
,
必须在集合元素中也加入
<cache>
子元素
而
Hibernate
仅仅是把与当前持久对象关联的对象的
OID
存放到缓存中。
如果希望把整个关联的对象的所有数据都存入缓存
,
则要在相应关联的对象的映射文件中配置
<cache>
元素
-->
<
cache
usage
=
"read-write"
/>
<
key
column
=
"categoryId"
not-null
=
"true"
/>
<
one-to-many
class
=
"org.qiujy.domain.cachedemo.Product"
/>
</
set
>
</
class
>
</
hibernate-mapping
>
|
<?
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
>
<!--
配置版本号
,
必须紧跟在
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
>
|
<
ehcache
>
<
diskStore
path
=
"c://ehcache/"
/>
<
defaultCache
maxElementsInMemory
=
"10000"
eternal
=
"false"
timeToIdleSeconds
=
"120"
timeToLiveSeconds
=
"120"
overflowToDisk
=
"true"
/>
<!--
设置
Category
类的缓存的数据过期策略
-->
<
cache
name
=
"org.qiujy.domain.cachedemo.Category"
maxElementsInMemory
=
"100"
eternal
=
"true"
timeToIdleSeconds
=
"0"
timeToLiveSeconds
=
"0"
overflowToDisk
=
"false"
/>
<!--
设置
Category
类的
products
集合的缓存的数据过期策略
-->
<
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
>
|
元素或属性
|
描述
|
<diskStore>
|
设置缓存数据文件的存放目录
|
<defaultCache>
|
设置缓存的默认数据过期策略
|
<cache>
|
设定具体的命名缓存的数据过期策略
每个命名缓存代表一个缓存区域,每个缓存区域有各自的数据过期策略。命名缓存机制使得用户能够在每个类以及类的每个集合的粒度上设置数据过期策略。
|
cache元素的属性
|
|
name
|
设置缓存的名字
,
它的取值为类的全限定名或类的集合的名字
|
maxInMemory
|
设置基于内存的缓存中可存放的对象最大数目
|
eternal
|
设置对象是否为永久的
,true
表示永不过期
,
此时将忽略
timeToIdleSeconds
和
timeToLiveSeconds
属性
;
默认值是
false
|
timeToIdleSeconds
|
设置对象空闲最长时间
,
超过这个时间
,
对象过期。当对象过期时
,EHCache
会把它从缓存中清除。
如果此值为
0,
表示对象可以无限期地处于空闲状态。
|
timeToLiveSeconds
|
设置对象生存最长时间
,
超过这个时间
,
对象过期。
如果此值为
0,
表示对象可以无限期地存在于缓存中。
|
overflowToDisk
|
设置基于内在的缓存中的对象数目达到上限后
,
是否把溢出的对象写到基于硬盘的缓存中
|
package org.qiujy.test.cache;
import java.util.List;
import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.qiujy.common.HibernateSessionFactory;
import org.qiujy.domain.cachedemo.Product;
public class TestCache {
public static void main(String[] args) {
//test cache.........
Session session2 = HibernateSessionFactory.getSession();
Transaction tx2 =null;
try{
tx2 = session2.beginTransaction();
List list = session2.createQuery("from Product").list();
for(int i = 0 ; i < list.size(); i++){
Product prod = (Product)list.get(i);
System.out.println(prod.getName());
}
tx2.commit();
}catch(HibernateException e){
if(tx2 != null){
tx2.rollback();
}
e.printStackTrace();
}finally{
HibernateSessionFactory.closeSession();
}
//-------------------
Session session3 = HibernateSessionFactory.getSession();
Transaction tx3 =null;
try{
tx3 = session3.beginTransaction();
Product prod = (Product)session3.get(Product.class, new Long(1));
System.out.println("从cache中得到,不执行SQL---" + prod.getName());
tx3.commit();
}catch(HibernateException e){
if(tx3 != null){
tx3.rollback();
}
e.printStackTrace();
}finally{
HibernateSessionFactory.closeSession();
}
}
}
|
<!--
设置默认的查询缓存的数据过期策略
-->
<
cache
name
=
"org.hibernate.cache.StandardQueryCache"
maxElementsInMemory
=
"50"
eternal
=
"false"
timeToIdleSeconds
=
"3600"
timeToLiveSeconds
=
"7200"
overflowToDisk
=
"true"
/>
<!--
设置时间戳缓存的数据过期策略
-->
<
cache
name
=
"org.hibernate.cache.UpdateTimestampsCache"
maxElementsInMemory
=
"5000"
eternal
=
"true"
overflowToDisk
=
"true"
/>
<!--
设置自定义命名查询缓存
customerQueries
的数据过期策略
-->
<
cache
name
=
"myCacheRegion"
maxElementsInMemory
=
"1000"
eternal
=
"false"
timeToIdleSeconds
=
"300"
timeToLiveSeconds
=
"600"
overflowToDisk
=
"true"
/>
|
<!--
启用查询缓存
-->
<
property
name
=
"cache.use_query_cache"
>
true
</
property
>
|
package org.qiujy.test.cache;
import java.util.List;
import org.hibernate.HibernateException;
import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.qiujy.common.HibernateSessionFactory;
import org.qiujy.domain.cachedemo.Product;
public class TessQueryCache {
public static void main(String[] args) {
Session session = HibernateSessionFactory.getSession();
Transaction tx =null;
try{
tx = session.beginTransaction();
Query query = session.createQuery("from Product");
//
激活查询缓存
query.setCacheable(true);
//
使用自定义的查询缓存区域
,
若不设置
,
则使用标准查询缓存区域
query.setCacheRegion("myCacheRegion");
List list = query.list();
for(int i = 0 ; i < list.size(); i++){
Product prod = (Product)list.get(i);
System.out.println(prod.getName());
}
tx.commit();
}catch(HibernateException e){
if(tx != null){
tx.rollback();
}
e.printStackTrace();
}finally{
HibernateSessionFactory.closeSession();
}
}
}
|