Hibernate的缓存机制
Hibernate的缓存
为什么使用缓存?
**hibernate提供了两种缓存来提升程序的性能**
- 一级缓存
- 二级缓存(类缓存,集合缓存,查询缓存)
说到缓存,让我们先来理解下Hibernate的生命周期
瞬时状态:此状态的对象存储于JVM内存中,与数据库的数据无关。
eg: 下面的user1对象就是瞬时状态
`Session session=HibernateSessionFactory.getSession();User user = new User(1, “812135023”, “一缕阳光”, ‘男’, 21, true, “本科”);`
- 持久状态:此状态的对象由hibernate的框架管理,数据库存在与hibernate内存中,数据可由hibernate持久化(保存)到数据库中。
eg:下面的user对象经过save方法就是持久化状态session.save(user); //此操作会导致数据持久化到数据库中
- 游离(脱管)状态:如果处于持久化的对象,其会话Session关闭后,此对象就会转变成游离(脱管)状态。
eg:经过close方法此时的user的对象处于游离(脱管)状态。session.close();
Hibernate实体对象状态间的转换示意图
一级缓存
一级缓存的使用周期相当短暂,其实,当我们采用save或get方法等等,我们已经把对象user加入到了一级缓存,当我们执行session.close();方法后,user对象,也会从一级缓存中清除出去。
eg:
Session session=HibernateSessionFactory.getSession();
User user = new User(1, "812135023", "一缕阳光", '男', 21, true, "本科");
session.save(user);//执行后,对象user已经加入到一级缓存
session.close(); //执行后,user会从一级缓存中清除出去
一级缓存的管理
hibernate提供了clear和evict,contains等3个方法来管理一级缓存。
session.evict(user);//从一级缓存中清除单个指定对象
session.clear();//清除一级缓存内的所有对象
boolean flag=session.contains(user);//检测一级缓存内是否存在某个对象
一级缓存代码演示
Session session=HibernateSessionFactory.getSession();
①User user=(User)session.get(User.class,20);
②User user2=(User)session.get(User.class,20);
session.close;
**配置hibernate配置文件show_sql属性为true后。如果看到一条执行语句,证明测试成功。
①执行的时候,会先查看一级缓存内部是否有要查询的20号对象,如果没有,发送sql语句到数据库查询,然后把此结果对象返回,并且存储于一级缓存中。
当执行②号语句时,因为①语句已经把数据对象放入一级缓存内,所有hibernate会直接从一级缓存内读取数据,从而提升程序性能。
二级缓存(外置缓存)
hibernate要查询的数据在一级缓存中,没找到,如果配置了二级缓存了。就会在二级缓存内继续查询。如果2级缓存中没有就会发送sql语句查询数据库。
因为二级缓存产品有好几种。
本文介绍如果使用hibernate自带的ehcache缓存
步骤:
- 2级缓存的jar包
- Hibernate配置文件中设置开启二级缓存
- 二级缓存本身的配置文件
- 配置实体类对二级缓存的配置
1.jar包,因为hibernate自带的,所以嘛,jar包hibernate里面就有
就是ehcache-1.2.3.jar。
以及struts自带的commons-logging-1.0.4.jar
2.Hibernate配置文件中设置开启二级缓存
<!-- 配置是否开启2级缓存 -->
<property name="cache.use_second_level_cache">true</property>
<!-- 配置2级缓存产品的Hibernate接口类 -->
<property name="cache.provider_class">org.hibernate.cache.EhCacheProvider</property>````
3.在src下创建ehcache.xml文件(文件名必须是ehcache(不然要改配置才能-name属性划分区域给User对象,默认区域
<?xml version="1.0" encoding="UTF-8"?>
<ehcache>
<cache maxElementsInMemory="10000" eternal="false" name="User" timeToIdleSeconds="120" timeToLiveSeconds="120" overflowToDisk="false" diskPersistent="false" diskExpiryThreadIntervalSeconds="120" memoryStoreEvictionPolicy="LRU" />
<defaultCache maxElementsInMemory="10000" eternal="false" timeToIdleSeconds="120" timeToLiveSeconds="120" overflowToDisk="true" diskPersistent="false" diskExpiryThreadIntervalSeconds="120" memoryStoreEvictionPolicy="LRU" />
</ehcache>
在实体类的配置文件中配置二级缓存区域属性
<class name="User" table="User" schema="SCOTT">
<!-- 配置2级缓存,存储的区域以及是否支持读和取 -->
<cache usage="read-only" region="User"/>
<id name="id" type="java.lang.Byte">
<column name="id" precision="2" scale="0" />
<generator class="assigned"></generator>
</id>
.....
</class>
这样就配置好了二级缓存类 。配置集合缓存只需要在此基础上, 在User.hbm.xml中的se标签内配置以下 属性
<set name="users" inverse="true" lazy="false">
<cache usage="read-only" region="User"/>
...省略其它配置
</set>
如何能看出来呢,当我们面对1对多的关系对象时候。调用对象的集合类型。此时。查到的数据就会存储到二级缓存中。
…
Set<Emp> emps=d.getEmps();//此时数据从数据库取出,放入二级缓存区域
至此集合缓存使用介绍结束。
最后一点,查询缓存
步骤
- 首先已经配置了2 级缓存
- 告诉hibernate配置的查询缓存
3 .使用query.setCacheable(true);方法
使用查询缓存,的前提是,我们要先配置二级缓存。然后在hibernate的配置文件下配置以下属性
<property name="cache.use_query_cache">true</property>
使用quer.settCacheable(true);方法,将不自动放 缓存的数据对象,放入到二级缓存区域。
/** * 配置查询缓存 * 1.首先已经配置了2级缓存 * 2.告诉hibernate配置的查询缓存 * <property name="cache.use_query_cache">true</property> * * 3.使用query.setCacheable(true);方法 * * */
Session session=HibernateSessionFactory.getSession();
String hql="from Emp e";
Query query=session.createQuery(hql);
/** * 第一次执行这一句话: * true:表明要使用查询缓存-sql语句查到的数据要放入2级缓存区 * * false:表明不要使用查询缓存-sql语句查到的数据仅仅当时使用-不放入缓存区域 * */
query.setCacheable(true);//通过这一句将查询到的数据加入缓存
List<Emp> emps=query.list();
HibernateSessionFactory.closeSession();
session=HibernateSessionFactory.getSession();
query = session.createQuery(hql);
/** * 第二次执行这一句话: * true:表明要使用查询缓存-先看缓存区域是否已经存在此数据,如果没有-就执行sql语句查询,然后放入缓存区 * * false:表明不要使用查询缓存-直接执行sql语句到数据库去查询数据 * */
query.setCacheable(false);
emps = query.list();
HibernateSessionFactory.closeSession();
//看到一行sql语句表明执行成功
查询缓存的代码演示和介绍到此结束,如果对本文有所异议,欢迎指导。
还没有评论,来说两句吧...