Spring cache的使用

spring cache是一个简单灵活的缓存框架

简单 : 基于注解,在需要缓存的方法上打上注解就可以缓存方法返回值

灵活 : 可自定义键值如何序列化,TTL等,更换缓存只需要注入相应的CacheManager就可以,同时spring已经提供了一些常用的缓存工具(RedisCacheManager,JCacheCacheManager,EhCacheCacheManager)

CacheManager

/** 可以定义多个CacheManager,在使用时进行选择
 * 不过需要指定一个默认的,不然启动会报错
 * @param redisConnectionFactory 
 * @return
 */
@Primary
@Bean("permission")
public CacheManager permissionCacheManager(RedisConnectionFactory redisConnectionFactory) {
    RedisCacheConfiguration cacheConfiguration = RedisCacheConfiguration.defaultCacheConfig()
            //前缀处理,也就是指定的缓存名称将以什么姓氏存到redis
            .computePrefixWith(cacheName -> "permission" + ":" + cacheName + ":")
            //键的序列化
            .serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(keySerializer()))
            //值的序列化
            .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(valueSerializer()))
            //是否缓存空值
            .disableCachingNullValues()
            //过期时间
            .entryTtl(Duration.ofDays(1));

    return RedisCacheManager.RedisCacheManagerBuilder
            .fromCacheWriter(RedisCacheWriter.nonLockingRedisCacheWriter(redisConnectionFactory))
            .cacheDefaults(cacheConfiguration)
            .build();
}

缓存注解

  • @Cacheable 最常用的一个,若缓存中没有对应键,则执行方法,并把返回值放入缓存,如果有则从缓存中取,不执行方法
  • @CacheEvict 删除缓存
  • @CachePut 修改缓存
  • @Caching 同时使用上面多个的时候使用
  • @CacheConfig 放在类上,可统一指定该类上的其它注解的一些设置(

@Cacheable

@Cacheable与@CachePut注解有相同的字段

如下是一个查找Menu对象的方法

  • cacheNames 缓存名称,默认为空串

  • cacheManager 指定cacheManager若不指定则使用默认的cacheManager,这里使用了上面定义的 permission

  • key 缓存的键,可使用字符串,入参,以及 #root root里有caches,method,args,target,targetClass可供使用

  • condition 条件,只用满足条件注解才生效,如下只有参数menuId==2时才生效

  • unless 与condition 类似,不过它可以处理返回值,如下只有返回的对象(#result)中的enable属性为1才生效,但是需要注意如果使用返回值中的某一个属性进行比较,但返回值是一个null,就会报错,unless是除非的意思,也就是说当unless中的条件满足时注解不生效

由于指定的permission cacheManager定义了前缀处理,所以存到resis里的键为

permission :Menu: one: selectOne: 2 只有当入参menuId=2并且返回值中的enable不等于1时注解才生效

@Cacheable(
        cacheNames = "Menu",
        cacheManager = "permission",
        key = "'one:'+#root.methodName+':'+#menuId",
        unless = "#result.enable == 1",
        condition = "#menuId==2")
public Menu selectOne(int menuId) {
    Menu menu = menuRepository.findById(menuId).orElse(null);
    System.out.println(menu);
    return menu;
}

@CacheEvict

有两个特殊的属性

  • allEntries=true 删除同一个cache(cacheName相同)下的所有键
  • beforeInvocation=true 无论方法有没有执行成功都删除缓存
@CacheEvict(
        cacheManager = "permission",
        cacheNames = "Menu", 
        allEntries = true, 
        beforeInvocation = true)
public void delete(int menuId) {
    if (true) {
        throw new RuntimeException();
    }
    menuRepository.deleteById(menuId);
}

@Caching

同时使用多个缓存注解

@Caching(put = @CachePut(key = "#menu.id"),
         evict = {@CacheEvict(key = "'all'"),
                @CacheEvict(key = "'enabled'", condition = "#menu.enable == 1")})
public Menu save(Menu menu) {
    return menuRepository.save(menu);
}

@CacheConfig

放在类上统一类中的其它缓存注解,如下相当于在这个类中的其它所有缓存注解都被默认加上了cacheNames = “Menu”,cacheManager = “permission”

@CacheConfig(cacheNames = "Menu", cacheManager = "permission")
public class MenuService {
......

KeyGenerator

某些业务不必每个cache都手动指定key,我们可以定义一定的策略自动生成key, 首先定义一个KeyGenerator放到spring容器中, KeyGenerator接口的generate方法有三个参数,当前对象,注解所在方法,方法的入参,如下定义一个简单的由类名+方法名+入参生成的key

@Bean("myKeyGenerator")
public KeyGenerator menuKeyGenerator() {
    return (target, method, params) -> {
        String baseName = target.getClass().getSimpleName() + ":" + method.getName();
        if (params.length == 0) {
            return baseName;
        }
        baseName = baseName + ":";
        for (Object param : params) {
            baseName = baseName + param.toString() + ",";
        }
        return baseName;
    };
}

使用 KeyGenerator

注意注解中的KeyGenerator与key是互斥的,两者只能生效一个

@Cacheable(keyGenerator = "myKeyGenerator")
public Menu select(int menuId) {
    return menuRepository.findById(menuId).orElse(null);
}