您的位置:首页 > 编程语言 > Java开发

为Spring集成的Hibernate配置二级缓存

2011-03-20 13:28 549 查看
在不少的项目中,也使用到了Hibernate的二级缓存,现在学习一下在Hibernate里面该如何使用二级缓存,先要把以下的配置信息加到beans.xml里的相应位置:

hibernate.cache.use_second_level_cache=true
hibernate.cache.use_query_cache=false
hibernate.cache.provider_class=org.hibernate.cache.EhCacheProvider

beans.xml

Xml代码

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
<context:annotation-config/>
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="org.gjt.mm.mysql.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/springjdbc?useUnicode=true&characterEncoding=UTF-8"/>
<property name="username" value="root"/>
<property name="password" value="456"/>
<!-- 连接池启动时的初始值 -->
<property name="initialSize" value="1"/>
<!-- 连接池的最大值 -->
<property name="maxActive" value="500"/>
<!-- 最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 -->
<property name="maxIdle" value="2"/>
<!-- 最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请 -->
<property name="minIdle" value="1"/>
</bean>

<!--通过这个配置就可以定义一个sessionFactory,这个对象在容器里面只存在一个,它是一个单例的形式-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource" ref="dataSource"/> <!--数据源-->
<property name="mappingResources">
<list>
<value>cn/itcast/bean/Person.hbm.xml</value><!--实体bean的映射元数据-->
</list>
</property>
<property name="hibernateProperties">
<value>
hibernate.dialect=org.hibernate.dialect.MySQL5Dialect
hibernate.hbm2ddl.auto=update<!--代表要不要根据映射元数据来生成数据库表结构-->
hibernate.show_sql=false <!--是否打印Hibernate执行的sql-->
hibernate.format_sql=false <!--是否要对它进行格式化-->
<!--这两个主要在测试阶段比较有用-->

<!--代表使用Hibernate的二级缓存-->
hibernate.cache.use_second_level_cache=true
<!--代表是否使用查询缓存,这里不使用,因为一般而言查询缓存的命中率并不是很高,所以我们没有
必要为每一个用户的查询缓存它的数据,所以这里设为false-->
hibernate.cache.use_query_cache=false
<!--用于指定使用缓存产品的驱动类-->
hibernate.cache.provider_class=org.hibernate.cache.EhCacheProvider
</value>
</property>
</bean>

<!--配置事务管理,使用的事务管理器是Spring为我们提供的,针对Hibernate的一个事务管理器-->
<!--只要是通过sessionFactory对象创建的session都会纳入到这个事务管理器中-->
<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>

<!--配置事务采用的申明方式,事务申明的方式有两种:1种是基于XML的方式,1种是基于注解的方式
,这里是使用注解方式来申明事务-->
<!--这段打开了对@Transaction注解的支持,这里用到的事务管理器就是前面提到的txManager-->
<tx:annotation-driven transaction-manager="txManager"/>

<bean id="personService" class="cn.itcast.service.impl.PersonServiceBean"/>
</beans>

ehcache这个缓存产品要使用到一个jar文件,是hibernate核心安装包下的:lib/optional/ehcache-1.2.3.jar,Ehcache底下可以定义一个配置文件,Ehcache默认的配置文件ehcache.xml(放在类路径下)

ehcache.xml
Xml代码

<?xml version="1.0" encoding="UTF-8"?>
<!--
defaultCache节点为缺省的缓存策略
maxElementsInMemory 内存中最大允许存在的对象数量
eternal 设置缓存中的对象是否永远不过期
overflowToDisk 把溢出的对象存放到硬盘上(当对象达到1000个的时候,是否会把溢出的【比如1001个】放到硬盘上去)
timeToIdleSeconds 指定缓存对象空闲多长时间就过期,过期的对象会被清除掉
timeToLiveSeconds 指定缓存对象总的存活时间
diskPersistent 当jvm结束是是否持久化对象(当缓存应用关闭的时候是否要把缓存的对象持久化到磁盘上去?)
diskExpiryThreadIntervalSeconds 指定专门用于清除过期对象的监听线程的轮询时间
-->
<ehcache>
<diskStore path="D:/cache"/><!--缓存的对象存在硬盘的哪个路径底下-->

<!--defaultCache 定义缓存的一些默认行为-->
<defaultCache maxElementsInMemory="1000" eternal="false" overflowToDisk="true"
timeToIdleSeconds="120"
timeToLiveSeconds="180"
diskPersistent="false"
diskExpiryThreadIntervalSeconds="60"/>

</ehcache>

这是缓存的默认的配置,配置好了这个缓存之后,就可以应用到hibernate里面的实体bean了,我们在需要使用缓存的实体bean的映射元数据配置里面添上缓存配置,

Person.hbm.xml

Hbm代码

<?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 package="cn.itcast.bean">
<class name="Person" table="person">
<cache usage="read-write" region="cn.itcast.bean.Person"/>
<!--
usage:设置缓存的策略,这里使用read-write。两个并发的事务可以对对象进行read,
但如果是当一个事务对它write的话,另一个事务是不能对它进行read的
region:指定缓存的区域名(可以定义为实体类的全称),在这个区域名里面存放缓存的对象。
-->
<id name="id">
<generator class="native"/>
</id>
<property name="name" length="10" not-null="true"/>
</class>
</hibernate-mapping>

这样的话,我们就可以为Person实体bean应用上缓存了,当然我们也可以为cn.itcast.bean.Person这个缓存域来定义一些它的特殊缓存设置,如果不定义的话,就默认使用ehcache.xml里面的<defaultCache/>缓存策略。如果有特别的缓存设置,可以对它进行定义

ehcache.xml

Xml代码

<?xml version="1.0" encoding="UTF-8"?>
<!--
defaultCache节点为缺省的缓存策略
maxElementsInMemory 内存中最大允许存在的对象数量
eternal 设置缓存中的对象是否永远不过期
overflowToDisk 把溢出的对象存放到硬盘上(当对象达到1000个的时候,是否会把溢出的【比如1001个】放到硬盘上去)
timeToIdleSeconds 指定缓存对象空闲多长时间就过期,过期的对象会被清除掉
timeToLiveSeconds 指定缓存对象总的存活时间
diskPersistent 当jvm结束是是否持久化对象(当缓存应用关闭的时候是否要把缓存的对象持久化到磁盘上?)
diskExpiryThreadIntervalSeconds 指定专门用于清除过期对象的监听线程的轮询时间
-->
<ehcache>
<diskStore path="D:/cache"/><!--缓存的对象存在硬盘的哪个路径底下-->

<!--defaultCache 定义缓存的一些默认行为-->
<defaultCache maxElementsInMemory="1000" eternal="false" overflowToDisk="true"
timeToIdleSeconds="120"
timeToLiveSeconds="180"
diskPersistent="false"
diskExpiryThreadIntervalSeconds="60"/>
<cache name="cn.itcast.bean.Person" maxElementsInMemory="100" eternal="false"
overflowToDisk="true" timeToIdleSeconds="300" timeToLiveSeconds="600" diskPersistent="false"/>
</ehcache>

现在已经为Person实体应用上了缓存,我们怎样去检验它目前应用上的缓存呢?我们应用上的缓存可以是这样的:如果一旦缓存里面存在某个id的对象后,当它第二次再去请求相同id的这个对象的时候,它就不会从数据库里获取数据的,而是从内存里面获取到这个缓存对象的,我们根据这点就可以来测试这个缓存是否起作用了?
我们的计划是这样的。。
getPerson(1)
//...把数据库关闭
getPerson(1)

getPerson(1),调用业务bean的getPerson(1)方法获取第一条记录,二级缓存就会把这条记录放到缓存里面去,就是说,当我们第二次再去得到1这个Person的话,它就会从内存里面获取,而不是从数据库里获取。
既然第二次获取是从内存里获取的,也就是说我们在中间把数据库关掉,当第二条getPerson(1)再获取1这个Person,如果说它能获取到,就证明这个对象是从内存获取的,因为这时候数据库已经关闭了,现在就做这么一个实验

PersonServiceTest.java

Java代码

package junit.test;

import java.util.List;

import org.junit.BeforeClass;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import cn.itcast.bean.Person;
import cn.itcast.service.PersonService;

public class PersonServiceTest {
private static PersonService personService;

@BeforeClass
//这个方法是在当单元测试PersonServiceTest实例被构建出来后就会执行
//可以在这个方法里面做一些初始化的操作
public static void setUpBeforeClass() throws Exception {
try {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("beans.xml");
personService = (PersonService)applicationContext.getBean("personService");
} catch (RuntimeException e) {
e.printStackTrace();
}
}

.........................................

@Test
public void testGetPerson() {
Person person = personService.getPerson(2);
System.out.println(person.getName());
try {
System.out.println("请关闭数据库");
Thread.sleep(15*1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("第二次开始获取");
person = personService.getPerson(2);
System.out.println(person.getName());
}

.........................................

}
这个实验主要是证明:当关闭数据库之后,我们再去获取Person2记录,如果它能返回并且正确打印信息,就证明这数据是从缓存(内存)里获取到这对象的,不是从数据库获取对象的,数据库person表如图:

首先要把D:/cache的数据清掉,执行下这个单元测试代码的testGetPerson()方法,其中要手动关闭Mysql数据库,控制台打印出:
李四
请关闭数据库
第二次开始获取
李四

说明:在我们数据库关闭的情况下,它也能获取到person name,也就证明这时候这个对象是从缓存里面获取的。
那么我们二级缓存的应用就已经成功了。二级缓存在企业中也是被大量使用到,所以大家要掌握。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: