一、缓存的概念:
以空间换时间;
二、Hibernate缓存的分类:
前面我们讲的缓存都是session缓存;也叫一级缓存;get,load等缓存都是内置的,一级缓存;
SessionFactory缓存,二级缓存;
这里以Class —— Student,1对多的关系(单向)来例子说明下:
Class.java:
package com.cy.model;public class Class { private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
Student.java:
package com.cy.model;import com.cy.model.Class;public class Student { private int id; private String name; private Class c; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Class getC() { return c; } public void setC(Class c) { this.c = c; } }
Class.hbm.xml:
Student.hbm.xml:
StudentTest.java:
package com.cy.service;import org.hibernate.Session;import org.hibernate.SessionFactory;import org.junit.After;import org.junit.Before;import org.junit.Test;import com.cy.model.Class;import com.cy.util.HibernateUtil;public class StudentTest { private SessionFactory sessionFactory=HibernateUtil.getSessionFactory(); private Session session; @Before public void setUp() throws Exception { session=sessionFactory.openSession(); session.beginTransaction(); } @After public void tearDown() throws Exception { session.getTransaction().commit(); session.close(); } /** * 同一个session中,同一个事务是有一级缓存的; */ @Test public void testCache1(){ Class c = (Class) session.get(Class.class, 1); System.out.println(c.getName()); Class c2 = (Class) session.get(Class.class, 1); System.out.println(c2.getName()); System.out.println(c == c2); /* * Hibernate: select class0_.classId as classId1_0_0_, class0_.className as classNam2_0_0_ from t_class class0_ where class0_.classId=? 08计算机本科 08计算机本科 true 第一次取c1的时候,已经将取出来的数据c1放到缓存中。 第二次取c2的时候,是有session缓存的。首选从缓存中找,找到,直接从缓存中拿,而不到数据库中去取了。 两个是同一个OID对象; */ } /** * 在不同的事务中,是否有缓存? */ @Test public void testCache2(){ Session session1 = sessionFactory.openSession(); session1.beginTransaction(); Class c = (Class) session1.get(Class.class, 1); System.out.println(c.getName()); session1.getTransaction().commit(); session1.close(); Session session2 = sessionFactory.openSession(); session2.beginTransaction(); Class c2 = (Class) session2.get(Class.class, 1); System.out.println(c2.getName()); session2.getTransaction().commit(); session2.close(); System.out.println(c == c2); /* * Hibernate: select class0_.classId as classId1_0_0_, class0_.className as classNam2_0_0_ from t_class class0_ where class0_.classId=? 08计算机本科 Hibernate: select class0_.classId as classId1_0_0_, class0_.className as classNam2_0_0_ from t_class class0_ where class0_.classId=? 08计算机本科 false 虽然在同一个session中有缓存,但是这里新开辟了一个事务,发出两个sql 不同的事务是不能共用一级缓存的。 一级缓存只适用于同一个session,同一个事务; */ }}
三、二级缓存策略提供商:
配置ehcache二级缓存,让不同的session,不同的事务能共享数据。
要用到SessionFactory缓存了;
使用到的ehcache相关的jar包:
jar包在hibernate压缩包中:
hibernate-release-4.3.5.Final\lib\optional\ehcache
可以参考hibernate压缩包中的hibernate-ehcache的配置:
hibernate-release-4.3.5.Final\project\hibernate-ehcache\bin
hibernate-release-4.3.5.Final\project\hibernate-ehcache\bin\hibernate-config\ ehcache.xml 以及 hibernate.cfg.xml
代码实现:
1)ehcache.xml配置:
2)hibernate.cfg.xml中相关ehcache二级缓存的配置:
com.mysql.jdbc.Driver jdbc:mysql://localhost:3306/hibernate root root org.hibernate.dialect.MySQL5Dialect true update true org.hibernate.cache.ehcache.EhCacheRegionFactory true ehcache.xml
3)最后,Class.hbm.xml中配置使用cache:
4)测试代码:
/** * 配置了二级缓存ehcache后,再次进行测试: */ @Test public void testCache3(){ Session session1 = sessionFactory.openSession(); session1.beginTransaction(); Class c = (Class) session1.get(Class.class, 1); System.out.println(c.getName()); session1.getTransaction().commit(); session1.close(); Session session2 = sessionFactory.openSession(); session2.beginTransaction(); Class c2 = (Class) session2.get(Class.class, 1); System.out.println(c2.getName()); session2.getTransaction().commit(); session2.close(); System.out.println(c == c2); /* * Hibernate: select class0_.classId as classId1_0_0_, class0_.className as classNam2_0_0_ from t_class class0_ where class0_.classId=? 08计算机本科 08计算机本科 false 只有1条sql,说明在第2次查的时候,使用到了二级缓存。 */ }
四、什么数据适合放二级缓存中
一般这些信息都是只读;不能修改;