Hibernate旅程(八)Hibernate缓存机制–二级缓存
最后更新于:2022-04-01 14:50:02
Hibernate二级级缓存
上篇介绍了[Hibernate](http://blog.csdn.net/lovesummerforever/article/details/20997879)[一级缓存](http://blog.csdn.net/lovesummerforever/article/details/20997879),主要是session缓存,session生命周期结束,缓存也就结束。二级缓存相对于一级缓存来说是一个范围更广阔一些,就比你住的地方周围有多个小卖铺(session缓存),和一个大型超市,原料加工厂送货的时候送小卖铺一份的同时,必然送一份到超市。而给第二个小卖铺送一份的同时,也送给超市一份,这个超市就是我们的SessionFactory。hibernate二缓存的又称为“SessionFactory的缓存”缓存的生命周期和SessionFactory(线程安全,一个数据库对应一个,重量级)的生命周期一致,所以SessionFactory可以管理二级缓存。
下面来看session控制的二级缓存。
### 二级缓存配置
1、需要引入第三方的jar包,hibernate的Cglib.jar。
2、在缓存的配置文件来控制缓存,我们可以拷贝hibernate已有项目中的ehcache.xml配置文件到自己的项目中。通过这个文件,我们可以对二级缓存进行设置,例如缓存的时间,缓存的代销,缓存间隔多长时间自动被清掉,缓存超时间直接缓存到磁盘指定的位置上等设置。
3、在hibernate.cfg.xml文件中加入缓存产品提供商。
~~~
<propertyname="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property>
~~~
4、启用二级缓存,设置为true,默认是true,添加上有助于我们手动关闭和开启二级缓存。
~~~
<propertyname="hibernate.cache.use_second_level_cache">true</property>
~~~
5、指定哪些实体类使用二级缓存。
方法一:在对应实体的.hbm.xml文件中设置,<cacheusage="read-only"/>这样如果我们想要知道二百个映射文件哪些使用了二级缓存,就要查看二百个文件,所以我们可以把对实体加入二级缓存的策略放到hibernate.cfg.xml进行配置。也就是方法二。
方法二:处于好管理的目的我们把哪个实体使用二级缓存的配置放到hibernate.cfg.xml文件中。例如执行把student实体加入二级缓存策略。 <!--
指定Student使用二级缓存
-->
~~~
<class-cacheclass="com.bjpowernode.hibernate.Student"usage="read-only"/>
~~~
注意:缓存策略通常采用read-only和read-write
缓存原则;通常是读大于写的数据进行缓存。
### 开启二级缓存在两个session中使用两次load()进行查询
代码如下所示。
~~~
/**
* 开启二级缓存
*
* 在两个session中发load查询
*/
public voidtestCache1() {
Sessionsession = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
Studentstudent = (Student)session.load(Student.class, 1);
System.out.println("student.name=" +student.getName());
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
try {
session = HibernateUtils.getSession();
session.beginTransaction();
Studentstudent = (Student)session.load(Student.class, 1);
//不会发出查询语句,因为配置二级缓存,session可以共享二级缓存中的数据
//二级缓存是进程级的缓存
System.out.println("student.name=" +student.getName());
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
}
~~~
打印结果如下图所示。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-21_576908fa7ec3a.jpg)
由此可知,虽然我们使用的是两个session,但第二次查询的时候没有再向数据库发出查询语句。也就是第一次查询的时候放入session的同时也放入SessionFactory缓存中一份,而第二个session再查询的时候,就直接从二级缓存中取出就可以了。
### 开启二级缓存在两个session中使用两次get()进行查询。
代码就是把上述代码load方法改动为get方法。
运行结果如下所示。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-21_576908fa7ec3a.jpg)
同load一样,对于sessionFactory级别的缓存,第二次查询时就不再向数据库发出查询sql命令。
### 开启二级缓存,在两个session中发出load查询,采用SessionFactory管理二级缓存
代码如下所示。
~~~
/**
* 开启二级缓存
*
* 在两个session中发load查询,采用SessionFactory管理二级缓存
*/
public voidtestCache3() {
Sessionsession = null;
try {
session= HibernateUtils.getSession();
session.beginTransaction();
Studentstudent = (Student)session.load(Student.class, 1);
System.out.println("student.name=" +student.getName());
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
//管理二级缓存 (evict清除的意思)
//HibernateUtils.getSessionFactory().evict(Student.class);
HibernateUtils.getSessionFactory().evict(Student.class, 1);
try {
session= HibernateUtils.getSession();
session.beginTransaction();
Studentstudent = (Student)session.load(Student.class, 1);
//会发出查询语句,因为二级缓存中的数据被清除了
System.out.println("student.name=" +student.getName());
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
}
~~~
显示结果如下所示。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-21_576908fa981c3.jpg)
根据显示结果可知,在两个session中发出load查询,第一次放入session缓存同时放入了二级缓存sessionFactory中。在第二个session中,首先使用SessionFactory对二级缓存管理,使用evict()方法清除二级缓存,所以第二个session采用load查询时,需要重新向数据库发送sql命令。
### 一级缓存和二级缓存交互
代码如下所示。
~~~
public voidtestCache4() {
Sessionsession = null;
try {
session= HibernateUtils.getSession();
session.beginTransaction();
//禁止将一级缓存中的数据放到二级缓存中。缓存模式设置为忽略。
session.setCacheMode(CacheMode.IGNORE);
Studentstudent =(Student)session.load(Student.class, 1);
System.out.println("student.name=" + student.getName());
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
try {
session= HibernateUtils.getSession();
session.beginTransaction();
Studentstudent = (Student)session.load(Student.class, 1);
//会发出查询语句,因为禁止了一级缓存和二级缓存的交互,一级缓存没有放到二级缓存中。
System.out.println("student.name=" +student.getName());
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
}
~~~
显示结果如下所示。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-21_576908fa981c3.jpg)
从上图可以看出,首先我们开启session的事务,然后设置了当写入session缓存中的同时不写入二级缓存,session.setCacheMode(CacheMode.IGNORE),这样第二个session再查询的时候就不能从二级缓存中找到相应的内容,仍需向数据库发出sql请求。
### 大批量数据的添加。
当添加或更新大批量数据的时候,为了防止缓存中数据量过大,我们可以设置删除session一级缓存,如果此时开启二级缓存,那么也会同样再二级缓存中存放一份,这样的话很可能会导致缓存的溢出,所以对于大数据来说,如果放入了一级缓存中,就应该禁止再放入二级缓存中。
代码如下所示。
~~~
/**
* 大批量的数据添加
*/
public voidtestCache5() {
Sessionsession = null;
try {
session= HibernateUtils.getSession();
session.beginTransaction();
//禁止一级缓存和二级缓存交互
session.setCacheMode(CacheMode.IGNORE);
for (int i=0;i<100; i++) {
Studentstudent = new Student();
student.setName("张三" +i);
//一级缓存放一份,二级缓存也放一份。
session.save(student);
//每20条更新一次
if (i %20 == 0) {
session.flush();
//清除缓存的内容
//只清除了一级缓存中的内容,没有清除一级缓存中的内容。如果数据量过大,一样溢出。
//大于大数据量来说,如果配置了一级缓存的话,就应该禁止再放入二级缓存中。
session.clear();
}
}
session.getTransaction().commit();
}catch(Exceptione) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
}
~~~
控制台打印sql如下图所示。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-21_576908fa5c8c0.jpg)
### Hibernate二级缓存总结
从上可知,二级缓存时比一级缓存作用域更大,是在整个应用程序中。一级缓存是不能被卸载的,是必需的,不允许也无法卸载,它是事务范围的缓存。而二级缓存是可配置的,可卸载的,SessionFactory生命周期和应用程序整个过程对应,就有可能出现并发问题。
什么样的数据适合放到二级缓存中?
1、很少被修改的数据
2、不是很重要的,并允许出现偶尔的并发数据
3、不会被并发的数据
4、常量数据
什么样的数据适不合放到二级缓存中?
1、经常被修改的数据。
2、绝对不允许并发的数据。
3、与其他应用共享的数据。