Hibernate的缓存分为两个级别:
一级缓存:指的是Session的缓存,所有使用同一个Session通过get或load查询的数据,都会保存在缓存中,下次再查询时,先从缓存中查找,如果找到,则不会再查询数据库,当Session关闭时,自动销毁。一般开发中不会使用,因为Session需要关闭。
二级缓存:保存在SessionFactory中,因此可以多个Session共同使用,二级缓存需要第三方的支持。可以使用oscache、ehcache之类缓存支持。默认Hibernate中加入的为ehcache缓存,这些缓存数据在一定时间后或SessionFactory销毁时自动销毁。
一、在hibernate.cfg.xml中配置使用二级缓存
1、在hibernate.cfg.xml中配置使用二级缓存,配置缓存支持类
[img]image001.jpg [/img]
2、还需要配置使用缓存的设置
[img]image002.jpg [/img]
3、配置使用二级缓存
[img]image003.jpg [/img]
二、在需要使用缓存查询的映射文件中,加入使用缓存的配置
<cache usage="read-only"/>
<id name="id" type="java.lang.Integer">
<column name="id" />
<generator class="native" />
</id>
其中usage表示使用缓存的模式,默认为read-only,可以使用以下4种,其中第4种已经不使用了:
read-only:缓存为只读,缓存中的数据不可以被修改,在数据查询要求不严格时使用,而且数据库数据改变的不多。
read-write:可以读写的,当数据库中数据修改时,同时对缓存中的数据进行修改,数据准确性很高,但如果频繁修改,性能相对比较低。
nonstrict-read-write:不严格的可读写模式,不是当数据库中数据修改就立刻修改缓存中的数据,而是一定时间后,对数据库和缓存进行比较,如果有数据不同,则进行数据同步。
transactional:事务处理的方式,但目前read-write已经支持了事务操作,因此该模式已经不使用了。
三、在src下加入ehcache.xml配置文件
需要在src下加入ehcache.xml配置文件,该文件可以在hibernate开发包hibernate-3.2\etc中查找到
<ehcache>
<!-- 表示缓存文件所保存的临时路径,java.io.tmpdir表示JDK所临时保存文件的路径 -->
<diskStore path="java.io.tmpdir"/>
<!--
以下为默认缓存配置,如果没有进行单独的配置,使用该配置进行缓存的保存:
maxInMemory - 设置内存中可以创建pojo数据的最大数量(针对一个pojo)
eternal - 设置缓存中的数据是否永久存在.如果为true, 当超过时间后,数据也不销毁.
timeToIdleSeconds - 设置从缓存对象创建后经过多长时间自动销毁. 只有当eternal为false时才可以使用. 单位为秒
timeToLiveSeconds - 设置缓存对象经过一次查询后,多长时间不再被查询后自动销毁(闲置时间).
overflowToDisk - 如果为true,则当数据数量超出范围后(InMemory中的范围)后,是否保存到硬盘上(保存的位置在上面的diskStore中定义).
-->
<defaultCache
maxElementsInMemory="10000"
eternal="false"
timeToIdleSeconds="600"
timeToLiveSeconds="120"
overflowToDisk="true"
/>
<!--
也可以自定义某一个类所使用的设置
name表示该类的包.类名
-->
<cache name="org.liky.pojo.Area"
maxElementsInMemory="20000"
eternal="false"
timeToIdleSeconds="1200"
timeToLiveSeconds="300"
overflowToDisk="true"
/>
</ehcache>
四、修改实现类(使用二级缓存)
修改实现类,实现类中需要为query或criteria设置打开缓存查询属性
public List<Area> findAll() throws Exception {
// TODO Auto-generated method stub
Criteria c = this.instance.getConnnection().createCriteria(Area.class);
// 设置使用缓存
c.setCacheable(true);
return c.list();
}
五、测试说明
缓存只能在对单个类操作时使用,如果使用了级联查询,则无法对缓存进行处理。
一级缓存:指的是Session的缓存,所有使用同一个Session通过get或load查询的数据,都会保存在缓存中,下次再查询时,先从缓存中查找,如果找到,则不会再查询数据库,当Session关闭时,自动销毁。一般开发中不会使用,因为Session需要关闭。
二级缓存:保存在SessionFactory中,因此可以多个Session共同使用,二级缓存需要第三方的支持。可以使用oscache、ehcache之类缓存支持。默认Hibernate中加入的为ehcache缓存,这些缓存数据在一定时间后或SessionFactory销毁时自动销毁。
一、在hibernate.cfg.xml中配置使用二级缓存
1、在hibernate.cfg.xml中配置使用二级缓存,配置缓存支持类
[img]image001.jpg [/img]
2、还需要配置使用缓存的设置
[img]image002.jpg [/img]
3、配置使用二级缓存
[img]image003.jpg [/img]
二、在需要使用缓存查询的映射文件中,加入使用缓存的配置
<cache usage="read-only"/>
<id name="id" type="java.lang.Integer">
<column name="id" />
<generator class="native" />
</id>
其中usage表示使用缓存的模式,默认为read-only,可以使用以下4种,其中第4种已经不使用了:
read-only:缓存为只读,缓存中的数据不可以被修改,在数据查询要求不严格时使用,而且数据库数据改变的不多。
read-write:可以读写的,当数据库中数据修改时,同时对缓存中的数据进行修改,数据准确性很高,但如果频繁修改,性能相对比较低。
nonstrict-read-write:不严格的可读写模式,不是当数据库中数据修改就立刻修改缓存中的数据,而是一定时间后,对数据库和缓存进行比较,如果有数据不同,则进行数据同步。
transactional:事务处理的方式,但目前read-write已经支持了事务操作,因此该模式已经不使用了。
三、在src下加入ehcache.xml配置文件
需要在src下加入ehcache.xml配置文件,该文件可以在hibernate开发包hibernate-3.2\etc中查找到
<ehcache>
<!-- 表示缓存文件所保存的临时路径,java.io.tmpdir表示JDK所临时保存文件的路径 -->
<diskStore path="java.io.tmpdir"/>
<!--
以下为默认缓存配置,如果没有进行单独的配置,使用该配置进行缓存的保存:
maxInMemory - 设置内存中可以创建pojo数据的最大数量(针对一个pojo)
eternal - 设置缓存中的数据是否永久存在.如果为true, 当超过时间后,数据也不销毁.
timeToIdleSeconds - 设置从缓存对象创建后经过多长时间自动销毁. 只有当eternal为false时才可以使用. 单位为秒
timeToLiveSeconds - 设置缓存对象经过一次查询后,多长时间不再被查询后自动销毁(闲置时间).
overflowToDisk - 如果为true,则当数据数量超出范围后(InMemory中的范围)后,是否保存到硬盘上(保存的位置在上面的diskStore中定义).
-->
<defaultCache
maxElementsInMemory="10000"
eternal="false"
timeToIdleSeconds="600"
timeToLiveSeconds="120"
overflowToDisk="true"
/>
<!--
也可以自定义某一个类所使用的设置
name表示该类的包.类名
-->
<cache name="org.liky.pojo.Area"
maxElementsInMemory="20000"
eternal="false"
timeToIdleSeconds="1200"
timeToLiveSeconds="300"
overflowToDisk="true"
/>
</ehcache>
四、修改实现类(使用二级缓存)
修改实现类,实现类中需要为query或criteria设置打开缓存查询属性
public List<Area> findAll() throws Exception {
// TODO Auto-generated method stub
Criteria c = this.instance.getConnnection().createCriteria(Area.class);
// 设置使用缓存
c.setCacheable(true);
return c.list();
}
五、测试说明
缓存只能在对单个类操作时使用,如果使用了级联查询,则无法对缓存进行处理。