Ehcache二级缓存,查询缓存,分布式缓存总结

二级缓存和查询缓存都至关于一个map
二级缓存缓存的keyidvalue为实体对象。通常load(),iterate()使用到二级缓存,list()须要结合查询缓存使用。iterate()和list()区别以下:
iterate()不须要开启查询缓存,它首先发出一个sqlselect s.id from Student s去数据库把id属性列表取出来,而后再根据id列表一个一个load(),若是缓存有从缓存取,若是缓存没有就从数据库取:select s.id,s.name,s.classid from     Student s where s.id=?,取出后再存入二级缓存。Iterate总会发出取id列表的语句。
List()须要开启查询缓存,它首先发出一个sqlselect s.id,s.name,s.classid from Student s…”去数据库取出全部相关实体,并将这些实体存入二级缓存,将此sql语句及一些相关信息做为keyid列表做为值,第二次查询这条语句时就会去根据sql语句及相关信息去key里找,若是有就会把id列表取出一个一个load(),接下来就和iterate同样了。List通常只有第一次发发出取实体列表的语句,之后的id列表就会去查询缓存取id列表,不会再发出sql语句。
 
前提:执行同一hql语句,如:select s from Student s
1.  关闭查询缓存,开启二级缓存时:
第二次查询属性时iterate只会发出获取id列表的sqllist会发出和第一次同样的请求实体的sql
2.  开启查询缓存,开启二级缓存
第二次查询属性时iterate只会发出获取id列表的sqllist不发sql
以上说明iterate只和二级缓存有关,list和二级缓存和查询缓存都有关。
 
查询缓存的key是一个QueryKey(其属性以下),valueid集合。

  
  
           
  
  
  1. public class QueryKey implements Serializable { 
  2.    private final String sqlQueryString;//sql语句 
  3.    private final Type[] types; 
  4.    private final Object[] values; 
  5.    private final Integer firstRow;//要查询的起始数 
  6.    private final Integer maxRows;//要查询的个数 
  7.    private final Map namedParameters; 
  8.    private final EntityMode entityMode; 
  9.    private final Set filters; 
  10.    private final int hashCode; 
  11.    ...... 
果以上属性中有一个不一样,查询缓存就不会命中。使用list()时,若是是第一次,查询缓存取出实体列表,而后从实体列表中提取出id列表做为value。若是是查询缓存命中,就会根据QueryKey取出id列表,而后根据id列表去二级缓存中load()对象,若是二级缓存中没有哪一个对象,就会根据id去数据库查询,二级缓存中没有n个对象就会去数据库查询n次:select p from pojo p where id=?,所以就会有机会出现传说中n+1问题。
在查询缓存中,查询缓存缓存普通属性,即对于属性的查询,value值为要查询的属性列表,对于实体,value值为实体的id列表,而对于属性的查询因为从value中直接能够查出就不会用到二级缓存,但对于实体的查询,必须结合二级缓存使用。下面对此进行说明:
前提:执行同一hql语句,如:select s from Student sselect s.name from Student s
1.  开启查询缓存,关闭二级缓存时:
第二次查询属性时不会发出sql,第一次查询实体时会发出sql
2.  开启查询缓存,开启二级缓存
第二次查询属性,实体时都不发sql
 
 
对于Ehcache分布式缓存,好像查询缓存不能更新:
在同一系统中,当用Hibernate的方式修改表数据(save,update,delete等等),这时EhCache会自动把缓存中关于此表的全部缓存所有删除掉(这样能达到同步)。但对于两个以上系统部署在不一样机子上,并在他们之间配置了ehcache的分布式缓存,当A系统修改表数据(save,update,delete)后,b系统会更新查询缓存吗?
答:好像是不能的,即ehcache的分布式缓存对查询缓存无效。(我在项目中配置了Ehcache的分布式缓存,二级缓存能够生效,但查询缓存不能更新。是否是一个系统中的Hibernate不可以识别其余系统中hibernate是否进行了save,update,delete操做?而Ehcache也没对此进行整合,望知道的大牛给说一下。)
相关文章
相关标签/搜索