Hibernate的一、二级缓存
什么是缓存?
并不是指计算机的内存或者CPU的一二级缓存。
缓存是指为了降低应用程序对物理数据源访问的频次,从而提高应用程序的运行性能的一种策略(缓存的作用对hibernate而言就是如何更快速的检索数据库,从数据库拿到我们想要的东西)
Hibernate一级缓存
hibernate默认是开启一级缓存的,一级缓存存放在session上
public class TestHibernate {
public static void main(String[] args) {
SessionFactory sf = new Configuration().configure().buildSessionFactory();
Session s = sf.openSession();
s.beginTransaction();
System.out.println("log1");
Category c1 = (Category)s.get(Category.class, 1); //刚开始获取session中没缓存,要访问数据库
System.out.println("log2");
Category c2= (Category)s.get(Category.class, 1); //session中已经有缓存,不再访问数据库
System.out.println("log3");
s.getTransaction().commit();
s.close();
sf.close();
}
}
第一次通过id=1获取对象的时候,session中是没有对应缓存对象的,所以会在"log1"后出现sql查询语句。
第二次通过id=1获取对象的时候,session中有对应的缓存对象,所以在"log2"后不会出现sql查询语句。
Hibernate二级缓存
提出问题,如何解决?
有些常用的数据,在一个session中缓存以后,我们希望在其他session中能够直接使用,而不用再次缓存怎么办?
使用更高级别的二级缓存,即每个session共用的缓存。
没开启二级缓存的案例情况:
创建了两个Session;
在第一个Session里:
第一次获取id=1的Category 会执行SQL语句;
第二次获取id=1的Category,不会执行SQL语句,因为有一级缓存 ;
在第二个Session里:
获取id=1的Category,会执行SQL语句,因为在第二个Session,没有缓存该对象。 所以总共会看到两条SQL语句。
Hibernate二级缓存配置
-
添加对应的二级缓存jar包
-
hibernate.cfg.xml 中增加对二级缓存的配置
hibernate本身不提供二级缓存,都是使用第三方的二级缓存插件
这里使用的是 EhCache提供的二级缓存
<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<!-- Database connection settings -->
<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="connection.url">jdbc:mysql://localhost:3306/test?characterEncoding=GBK</property>
<property name="connection.username">root</property>
<property name="connection.password">admin</property>
<!-- SQL dialect -->
<property name="dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="current_session_context_class">thread</property>
<property name="show_sql">true</property>
<property name="hbm2ddl.auto">update</property>
<property name="hibernate.cache.use_second_level_cache">true</property> //对二级缓存的配置
<property name="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property> //对二级缓存的配置
<mapping resource="com/how2java/pojo/Product.hbm.xml" />
<mapping resource="com/how2java/pojo/Category.hbm.xml" />
<mapping resource="com/how2java/pojo/User.hbm.xml" />
</session-factory>
</hibernate-configuration>
- ehcache.xml
在src目录下,创建一个ehcache.xml用于EHCache的缓存配置
<ehcache>
<diskStore path="java.io.tmpdir"/>
<defaultCache
maxElementsInMemory="10000" //在内存中缓存的element(事务)的最大数目
eternal="false" //设定缓存的elements是否永远不过期。如果为true,则缓存的数据始终有效.
timeToIdleSeconds="120"
//当缓存在EhCache中的数据前后两次访问的时间超过timeToIdleSeconds的属性取值时,这些数据便会删除
timeToLiveSeconds="120" //缓存element的有效生命期,默认是0.,也就是element存活时间无穷大
overflowToDisk="true" //设定当内存缓存溢出的时候是否将过期的element缓存到磁盘上
/>
</ehcache>
- 设置HBM
对于要进行二级缓存的实体类,进行配置,增加< cache usage=“read-only” />
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.how2java.pojo">
<class name="Category" table="category_">
<cache usage="read-only" /> //表示开启二级缓存,且为只读模式
<id name="id" column="id">
<generator class="native">
</generator>
</id>
<property name="name" />
<set name="products" lazy="true">
<key column="cid" not-null="false" />
<one-to-many class="Product" />
</set>
</class>
</hibernate-mapping>
- 测试效果
使用不同的session,都去获取id=1的category,只会访问一次数据库。
因为第二次获取虽然没有从第二个session中拿到缓存,但是从sessionfactory中拿到了Category缓存对象
喜欢的朋友可以关注我的个人公众号,后台回复java资料可免费领取资源。