在之前,我们采用的进程缓存可以是ConcurrentHashMap,加了锁的LinkedHashMap,WeakHashMap,以及Google guava Cache,需要自己定义或者实现缓存过期处理以及LRU等。这些本地缓存有以下缺点:①和自己业务逻辑耦合度高。②难以整合其他的缓存方案。
spring3之后可以通过注解来配置缓存。通过很少的代码就可以提供缓存功能,让代码变得更优雅。
springframework.cache并不是一个具体的缓存方法,而是对缓存的一个抽象,支持ehcache和redis等专业的缓存方案,还可以通过Spring Expression Language(SpEL)来定义缓存的key和condition。
比较常用的注解有“ Cacheable”,“CacheEvict”,“CachePut”。
Cacheable:指明此方法的的结果,或者类的所有方法的结果可以被缓存,可以通过参数列表或者keyGenerator来指定缓存的key的生成规则,用SpEL来指定添加缓存的条件。
每次方法被执行,如果在缓存中发现了key,就返回缓存中相应的value,反之就会触发函数中的目标方法,并结果加入缓存。
CacheEvict:指明某个方法或者某个类的所有方法触发清除缓存。注意缓存中加上allEntries=true使其立即清除所有缓存。
CachePut: 指明某个方法或者某个类的所有方法触发添加缓存。与Cacheable相反,此注解总是触发目标方法,并添加到缓存
示例
@Cacheable(value = "usercache", keyGenerator="smpkeyGenerator")
public User getUser(String no,String name){
LogCore.BASE.debug("invoke persistent:{},{}", no, name);
return new User(no, name);
}
/**
* allEntries=true 是立即清除所有缓存
*/
@CacheEvict(value = "usercache", keyGenerator="smpkeyGenerator")
public boolean clearUser(String no,String name){
LogCore.BASE.debug("invoke clear:{},{}", no, name);
return true;
}
@CachePut(value = "usercache", keyGenerator="smpkeyGenerator")
public User putUser(String no, String name){
User usr = new User(no, name);
return usr;
}
keyGenerator示例:
@Bean
public KeyGenerator smpkeyGenerator() {
return (target, method, params)->{
StringBuilder sb = new StringBuilder();
sb.append(target.getClass().getSimpleName());
Stream.of(params).forEach(sb.append("_")::append);
return sb.toString();
};
}
SPEL表达式:
要想控制加入缓存的条件,可以使用SPEL表达式,可以用conditions和unless前者是对传入的参数进行筛选,后者可以对返回值进行筛选
例如:
@Cacheable(value = "usercache", keyGenerator = "smpkeyGenerator",condition="#name!=null", unless = "#result==null")
public User getUser(String no, String name) {
LogCore.BASE.debug("invoke getUser:{},{}", no, name);
if (Util.anyNonEmpty(no, name)) {
return new User(no, name);
}
return null;
}