您的位置:首页 > 其它

Hibernate的缓存机制

2015-09-19 00:00 633 查看
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缓存

步骤:

1. 2级缓存的jar包

2. Hibernate配置文件中设置开启二级缓存

3. 二级缓存本身的配置文件

4. 配置实体类对二级缓存的配置

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();//此时数据从数据库取出,放入二级缓存区域


至此集合缓存使用介绍结束。

最后一点,查询缓存

步骤

1. 首先已经配置了2 级缓存

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语句表明执行成功


查询缓存的代码演示和介绍到此结束,如果对本文有所异议,欢迎指导。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: