(一)什么样的数据适合存放到第二级缓存中?
1 很少被修改的数据
2 不是很重要的数据,允许出现偶尔并发的数据
3 不会被并发访问的数据
4 参考数据,指的是供应用参考的常量数据,它的实例数目有限,它的实例会被许多其他类的实例引用,实例极少或者从来不会被修改。
(二)不适合存放到第二级缓存的数据?
1 经常被修改的数据
2 财务数据,绝对不允许出现并发
3 与其他应用共享的数据。
在一个数据库系统中,如果缓存设置的合适,那么可以极大的提高系统的效率,Hibernate作为一个ORM工具
提供了缓存的机制,包括一级(Session级)缓存和二级(SessionFactory级)缓存。这里主要总结一下二级缓存。
1.首先需要在hibernate.cfg.xml中配置,当然需要导入缓存的jar包
<property name="hibernate.cache.use_query_cache">true</property> <property name="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property>
如果想缓存使用findall()、list()、Iterator()、createCriteria()、 createQuery()等方法获得的数据结果集。 hibernate.cache.use_query_cache必须配置
2.在每个实体的hbm文件中配置cache元素,usage可以是read-only或者是read-write等。
<hibernate-mapping package="com.hibernate.model"> <class name="TUser" table="t_user"> <cache usage="read-write"/> <id column="tno" name="tno"> <!-- 主键生成策略 --> <!-- <generator class="assigned" />--> <!-- <generator class="sequence"> <param name="sequence">seq_tuser_tno</param> </generator> --> <!-- <generator class="identity" />--> <generator class="native" /> </id> <property name="tname" column="tname" type="string" not-null="true"></property> <property name="tpwd" column="tpwd" type="string" not-null="true"></property> </class> </hibernate-mapping>
如果相对具体某个类的缓存进行特定的配置,需要在ehcache.xml进行配置:
<cache name="com.hibernate.model.TUser" maxElementsInMemory="100" eternal="true" timeToIdleSeconds="0" timeToLiveSeconds="0" overflowToDisk="false" />
3.Query或Criteria()时设置其setCacheable(true);
public void list(){ //读取配置文件 Configuration config = new Configuration().configure(); //创建sessionFactory对象 SessionFactory sessionFactory = config.buildSessionFactory(); //打开session Session session = sessionFactory.openSession(); //开始事务 Transaction tx = session.beginTransaction(); //需要antlr.jar包, HQL 到 SQL的转换 Query query = session.createQuery("from TUser"); query.setCacheable(true); List<TUser> userList = query.list(); for(TUser user:userList){ System.out.println("tno:"+user.getTno()+" tname:"+user.getTname()+" tpwd:"+user.getTpwd()); } //test secondlevel_session,观察是否还会发出sql语句 List<TUser> list = query.list(); for(TUser user:list){ System.out.println("tno:"+user.getTno()+" tname:"+user.getTname()+" tpwd:"+user.getTpwd()); } //关闭session session.close(); }
执行以上代码时,第一次会查询数据库,但是后面就直接从缓存中查询,而不会使用数据库的连接,提高了性能。
以上任一环节都不能少,比如cache元素没有配置,那么就会导致查询district的时候发起N个数据库的连接,这样会极大的降低性能。