文章目录
- 1. Hibernate的一级缓存
- 2. 证明一级缓存的存在
- 3. 一级缓存的内部结构
- 4. Hibernate的二级缓存
1. Hibernate的一级缓存
缓存是计算机领域非常通用的概念。它介于应用程序和永久性数据存储源(如硬盘上的文件或者数据库)之间,其作用是降低应用程序直接读写永久性数据存储源的频率,从而提高应用的运行性能。缓存中的数据是数据存储源中数据的拷贝,缓存的物理介质通常是内存。
Hibernate的缓存分为一级缓存和二级缓存,Hibernate的这两级缓存都位于持久化层,存储的都是数据库数据的备份。其中第一级缓存为Hibernate的内置缓存,不能被卸载。接下来围绕Hibernate的一级缓存进行详细的讲解。
Hibernate的一级缓存就是指Session缓存,Session缓存是一块内存空间,用来存放管理的Java对象。在使用Hibernate査询对象的时候,首先会使用对象属性的OID值在Hibernate的一级缓存中进行査找,如果找到匹配OID值的对象,就直接将该对象从一级缓存中取出使用,不会再查询数据库。如果没有找到相同OID值的对象,则会去数据库中査找相应数据。当从数据库中査询到所需数据时,该数据信息也会放置到一级缓存中。Hibernate的一级缓存的作用就是减少对数据库的访问次数。
在Session接口的实现中包含一系列的Java集合,这些Java集合构成了Session缓存。只要Session实例没有结束生命周期,存放在它缓存中的对象也不会结束生命周期,故一级缓存也被称为是Session基本的缓存。
Hibernate的一级缓存有如下特点:
- 当应用程序调用Session接口的save、update、saveOrUpdate时,如果Session缓存中没有相应的对象,Hibernate就会自动的把从数据库中查询到的相应对象信息加入到一级缓存中去。
- 当调用Session接口的load、get方法,以及Query接口的list、iterator方法时,会判断缓存中是否存在该对象,有则返回,不会査询数据库,如果缓存中没有要査询对象,再去数据库中查询对应对象,并添加到一级缓存中。
- 当调用Session的close方法时,Session缓存会被清空。
2. 证明一级缓存的存在
我们已经大致了解了什么是一级缓存了,那么一级缓存具体是否存在呢,我们可以通过如下的程序来证明一级缓存是存在的。
public void demo1 ()(
Configuration configuration = new Configuration().configure();
SessionFactory sessionFactory = configuration.buildSessionFactory();
Transaction transaction = session.beginTransaction();
Customer customer1 = session.get(Customer.class, 1L);//马上发送一条 SQL查询1号客户,并将数据存入一级缓存
System.out.println(customer1);
Customer customer2 = session.get(Customer.class, 1L);//没有发生 SQL语句,从一级缓存中获取数据
System.out.println(customer2);
System.out.println(customer1 = customer2);//true,一级缓存缓存的是对象的地址
transaction.commit();
session.close();
sessionFactory.close();
}
在以上代码中,第一次执行Session的get方法获取customer1对象时,由于一级缓存中没有数据,所以Hibernate会向数据库发送一条sql语句,查询id等于1的对象。当再次调用了Session的get方法获取customer2对象时,将不会再发送sql语句,这是因为customer2对象是从一级缓存中获取的。
3. 一级缓存的内部结构
之前的博文老王介绍过Hibernate的持久态对象能够自动更新数据库,其实就是依赖了一级缓存,那么一级缓存为什么就可以去更新数据库了呢,其实是因为一级缓存的一块特殊的区域就是快照区。
Hibernate向一级缓存放入数据时,同时复制一份数据放入到Hibernate快照中,当使用commit方法提交事务时,同时会清理Session的一级缓存,这时会使用OID判断一级缓存中的对象和快照中的对象是否一致,如果两个对象中的属性发生变化,则执行update语句,将缓存的内容同步到数据库,并更新快照。如果一致,则不执行update语句。Hibernate快照的作用就是确保一级缓存中的数据和数据库中的数据一致。
4. Hibernate的二级缓存
Hibernate提供了基于应用程序级别的缓存,即为二级缓存,可以跨多个session,即不同的session都可以访问缓存数据,这个缓存也叫二级缓存。
Hibernate提供的二级缓存有默认的实现,且是一种可插配的缓存框架。如果想用二级缓存,只需要在hibernate.cfg.xmI中配置即可,不想用,直接移除,不影响代码。如果觉得Hibernate提供的框架框架不好用,自己可以换其他的缓存框架或自己实现缓存框架都可以。开发过程中我们一般都不使用Hibernate提供的二级缓存,而使用性能更好的缓存,如Ehcache、Memcache或Redis等,所以老王在这里就只是简单的介绍下Hibernate二级缓存的配置。
Hibernate二级缓存的配置:
<!-- 开启二级缓存 -->
<property name="hibernate.cache.use_second_level_cache">true</property>
<!-- 指定使用哪一个缓存框架(默认提供的) -->
<property name="hibernate.cache.provider_class">org.hibernate.cache.HashtableCacheProvider</property>
<!-- 开启查询缓存 -->
<property name="hibernate.cache.use_query_cache">true</property>
<!-- 指定哪一些类需要加入二级缓存 -->
<class-cache usage="read-write" class="com.joker.domain.Customer"/>
<class-cache usage="read-only" class="com.joker.domain.Linkman"/>
<!-- 集合缓存(集合缓存的元素对象,也加加入二级缓存) -->
<collection-cache usage="read-write" collection="com.joker.domain.Customer.linkmans"/>
缓存策略:
<class-cache usage="read-only"/> <!-- 放入二级缓存的对象,只读 -->
<class-cache usage="nonstrict-read-write"/> <!-- 非严格的读写 -->
<class-cache usage="read-write"/> <!-- 读写,放入二级缓存的对象可以读、写 -->
<class-cache usage="transactional"/> <!-- 基于事务的策略 -->