标签:
一级缓存
|
二级缓存
|
|
存放数据的形式
|
相互关联的持久化对象
|
对象的散装数据
|
缓存的范围
|
事务范围,每个事务都拥有单独的一级缓存
|
进程范围或集群范围,缓存被同一个进程或集群范围内所有事务共享
|
并发访问策略
|
由于每个事务都拥有单独的一级缓存不会出现并发问题,因此无须提供并发访问策略
|
由于多个事务会同时访问二级缓存中的相同数据,因此必须提供适当的并发访问策略,来保证特定的事务隔离级别
|
数据过期策略
|
处于一级缓存中的对象永远不会过期,除非应用程序显示清空或者清空特定对象
|
必须提供数据过期策略,如基于内存的缓存中对象的最大数目,允许对象处于缓存中的最长时间,以及允许对象处于缓存中的最长空闲时间
|
物理介质
|
内存
|
内存和硬盘,对象的散装数据首先存放到基于内存的缓存中,当内存中对象的数目达到数据过期策略的maxElementsInMemory值,就会把其余的对象写入基于硬盘的缓存中
|
缓存软件实现
|
在Hibernate的Session的实现中包含
|
由第三方提供,Hibernate仅提供了缓存适配器,用于把特定的缓存插件集成到Hibernate中
|
启用缓存的方式
|
只要通过Session接口来执行保存,更新,删除,加载,查询,Hibernate就会启用一级缓存,对于批量操作,如不希望启用一级缓存,直接通过JDBCAPI来执行
|
用户可以再单个类或类的单个集合的粒度上配置第二级缓存,如果类的实例被经常读,但很少被修改,就可以考虑使用二级缓存,只有为某个类或集合配置了二级缓存,Hibernate在运行时才会把它的实例加入到二级缓存中
|
用户管理缓存的方式
|
一级缓存的物理介质为内存,由于内存的容量有限,必须通过恰当的检索策略和检索方式来限制加载对象的数目,Session的evit()方法可以显示的清空缓存中特定对象,但不推荐
|
二级缓存的物理介质可以使内存和硬盘,因此第二级缓存可以存放大容量的数据,数据过期策略的maxElementsInMemory属性可以控制内存中的对象数目,管理二级缓存主要包括两个方面:选择需要使用第二级缓存的持久化类,设置合适的并发访问策略;选择缓存适配器,设置合适的数据过期策略。SessionFactory的evit()方法也可以显示的清空缓存中特定对象,但不推荐
|
1 public class Client 2 { 3 public static void main(String[] args) 4 { 5 Session session = HibernateUtil.getSessionFactory().openSession(); 6 Transaction tx = null; 7 try 8 { 9 /*开启一个事务*/ 10 tx = session.beginTransaction(); 11 /*从数据库中获取id="402881e534fa5a440134fa5a45340002"的Customer对象*/ 12 Customer customer1 = (Customer)session.get(Customer.class, "402881e534fa5a440134fa5a45340002"); 13 System.out.println("customer.getUsername is"+customer1.getUsername()); 14 /*事务提交*/ 15 tx.commit(); 16 17 System.out.println("-------------------------------------"); 18 19 /*开启一个新事务*/ 20 tx = session.beginTransaction(); 21 /*从数据库中获取id="402881e534fa5a440134fa5a45340002"的Customer对象*/ 22 Customer customer2 = (Customer)session.get(Customer.class, "402881e534fa5a440134fa5a45340002"); 23 System.out.println("customer2.getUsername is"+customer2.getUsername()); 24 /*事务提交*/ 25 tx.commit(); 26 27 System.out.println("-------------------------------------"); 28 29 /*比较两个get()方法获取的对象是否是同一个对象*/ 30 System.out.println("customer1 == customer2 result is "+(customer1==customer2)); 31 } 32 catch (Exception e) 33 { 34 if(tx!=null) 35 { 36 tx.rollback(); 37 } 38 } 39 finally 40 { 41 session.close(); 42 } 43 } 44 }
结果 Hibernate: select customer0_.id as id0_0_, customer0_.username as username0_0_, customer0_.balance as balance0_0_ from customer customer0_ where customer0_.id=? customer.getUsername islisi ------------------------------------- customer2.getUsername islisi ------------------------------------- customer1 == customer2 result is true
sessionFactory.evict(Customer.class, new Integer(1));
evict(Class arg0) 将指定类的所有持久化对象从二级缓存中清除,释放其占用的内存资源。
sessionFactory.evict(Customer.class);
evictCollection(String arg0) 将指定类的所有持久化对象的指定集合从二级缓存中清除,释放其占用的内存资源。
sessionFactory.evictCollection("Customer.orders");
<!-- EHCache的配置,hibernate.cfg.xml -->
<hibernate-configuration>
<session-factory>
<!-- 设置二级缓存插件EHCache的Provider类-->
<property name="hibernate.cache.provider_class">
org.hibernate.cache.EhCacheProvider
</property>
<!-- 启动"查询缓存" -->
<property name="hibernate.cache.use_query_cache">
true
</property>
</session-factory>
</hibernate-configuration>
<!-- ehcache.xml --> <?xml version="1.0" encoding="UTF-8"?> <ehcache> <!-- 缓存到硬盘的路径 --> <diskStore path="d:/ehcache"></diskStore> <!-- 默认设置 maxElementsInMemory : 在內存中最大緩存的对象数量。 eternal : 缓存的对象是否永远不变。 timeToIdleSeconds :可以操作对象的时间。 timeToLiveSeconds :缓存中对象的生命周期,时间到后查询数据会从数据库中读取。 overflowToDisk :内存满了,是否要缓存到硬盘。 --> <defaultCache maxElementsInMemory="200" eternal="false" timeToIdleSeconds="50" timeToLiveSeconds="60" overflowToDisk="true"></defaultCache> <!-- 指定缓存的对象。 下面出现的的属性覆盖上面出现的,没出现的继承上面的。 --> <cache name="com.suxiaolei.hibernate.pojos.Order" maxElementsInMemory="200" eternal="false" timeToIdleSeconds="50" timeToLiveSeconds="60" overflowToDisk="true"></cache> </ehcache>
<!-- *.hbm.xml --> <?xml version="1.0" encoding=‘UTF-8‘?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" > <hibernate-mapping> <class> <!-- 设置该持久化类的二级缓存并发访问策略 read-only read-write nonstrict-read-write transactional--> <cache usage="read-write"/> </class> </hibernate-mapping>
若存在一对多的关系,想要在在获取一方的时候将关联的多方缓存起来,需要在集合属性下添加<cache>子标签,这里需要将关联的对象的hbm文件中必须在存在<class>标签下也添加<cache>标签,不然Hibernate只会缓存OID。
<hibernate-mapping> <class name="com.suxiaolei.hibernate.pojos.Customer" table="customer"> <!-- 主键设置 --> <id name="id" type="string"> <column name="id"></column> <generator class="uuid"></generator> </id> <!-- 属性设置 --> <property name="username" column="username" type="string"></property> <property name="balance" column="balance" type="integer"></property> <set name="orders" inverse="true" cascade="all" lazy="false" fetch="join"> <cache usage="read-only"/> <key column="customer_id" ></key> <one-to-many class="com.suxiaolei.hibernate.pojos.Order"/> </set> </class> </hibernate-mapping>
标签:
原文地址:http://www.cnblogs.com/dullsky/p/5059935.html