二级缓存及外部缓存,它能够越过数个Session,通过同一个SessionFactory进行管理和维护。
接上例的代码:http://blog.csdn.net/kunshan_shenbin/archive/2008/09/03/2874375.aspx
要使用二级缓存,首先需要对hibernate.cfg.xml文件进行修改:
1.<?xml version="1.0" encoding="utf-8"?><br />2.<!DOCTYPE hibernate-configuration PUBLIC<br />3. "-//Hibernate/Hibernate Configuration DTD 3.0//EN"<br />4. "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd"><br />5.<br />6.<hibernate-configuration><br />7.<br />8. <session-factory><br />9.<br />10. ....<br />11. <property name="hibernate.cache.provider_class"><br />12. org.hibernate.cache.HashtableCacheProvider<br />13. </property><br />14. ....<br />15.<br />16. </session-factory><br />17.<br />18.</hibernate-configuration>
HashtableCache是Hibernate自己所提供的二级缓存的实现,但是由于其性能和功能有限,一般只用于开发和测试。
当然我们可以使用第三方的实现,譬如:org.hibernate.cache.EhCacheProvider,这个需要ehcathe.jar包。
修改User.hbm.xml文件:
1.<?xml version="1.0" encoding="utf-8"?><br />2.<!DOCTYPE hibernate-mapping<br />3. PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"<br />4. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"><br />5.<br />6.<hibernate-mapping><br />7.<br />8. <class name="com.hb3.pack_21.model.User" table="user"><br />9.<br />10. <cache usage="read-write"/><br />11.<br />12. <id name="id" column="id" type="java.lang.Integer"><br />13. <generator class="native"/><br />14. </id><br />15.<br />16. <property name="name" column="name" type="java.lang.String"/><br />17.<br />18. <many-to-one name="room"<br />19. column="room_id"<br />20. class="com.hb3.pack_21.model.Room"<br />21. cascade="save-update"<br />22. outer-join="true"/><br />23.<br />24. </class><br />25.<br />26.</hibernate-mapping>
这里追加的策略有read-only、read-write、nonstrict-read-write以及transactional。不同的第三方缓存实现所支持的种类是不同的,请参阅:http://www.hibernate.org/hib_docs/v3/reference/en/html/performance.html#performance-cache
在工程的classpath下追加ehcache.xml文件,内容为:
1.<ehcache><br />2.<br />3. <diskStore path="java.io.tmpdir"/><br />4.<br />5. <defaultCache<br />6. maxElementsInMemory="10000"<br />7. eternal="false"<br />8. timeToIdleSeconds="120"<br />9. timeToLiveSeconds="120"<br />10. verflowToDisk="true"<br />11. /><br />12.<br />13.</ehcache>
测试代码如下:
1.package com.hb3.pack_21;<br />2.<br />3.import java.io.IOException;<br />4.import java.sql.SQLException;<br />5.<br />6.import org.hibernate.Session;<br />7.import org.hibernate.SessionFactory;<br />8.import org.hibernate.cfg.Configuration;<br />9.<br />10.import com.hb3.pack_21.model.User;<br />11.<br />12.public class BusinessService {<br />13.<br />14. public static void main(String[] args) throws IOException, SQLException {<br />15.<br />16. Configuration config = new Configuration().configure();<br />17. SessionFactory sessionFactory = config.buildSessionFactory();<br />18. Session session = sessionFactory.openSession();<br />19.<br />20.<br />21. User user1 = (User) session.load(User.class, new Integer(1));<br />22. user1.getName();<br />23.<br />24. session.close();<br />25. session = sessionFactory.openSession();<br />26.<br />27. User user2 = (User) session.load(User.class, new Integer(1));<br />28. user2.getName();<br />29.<br />30.<br />31. session.close();<br />32. sessionFactory.close();<br />33. }<br />34.}
如果不使用二级缓存机制,生成的SQL文为:
Hibernate:select user0_.id as id0_, user0_.name as name0_0_, user0_.age as age0_0_from user user0_ where user0_.id=?Hibernate:select user0_.id as id0_, user0_.name as name0_0_, user0_.age as age0_0_from user user0_ where user0_.id=?
使用后,第二级做相同的查询会从二级缓存中读取。
当然,如果打算从二级缓存中清空对象的缓存信息,可以使用SessionFactory的evict()方法,如:
sessionFactory.evict(User.class, user.getId());
漫无目的的生活就像出海航行而没有指南针