Spring缓存@Cacheable 使用详解

Spring缓存@Cacheable 使用详解

  • 参考
  • 核心代码

参考

底层实现:https://juejin.cn/post/6959002694539444231

缺点:单机版本,非分布式。
可以使用内存作为cache也可以使用redis。

核心代码

配置:

@Configuration
public class CacheConfig implements CachingConfigurer {

    @Override
    @Bean
    public CacheManager cacheManager() {
        SimpleCacheManager simpleCacheManager = new SimpleCacheManager();

        final List<CaffeineCache> caches = new ArrayList<>();

        caches.add(new CaffeineCache(CacheConstants.CAFFEINE_1M_CACHE, Caffeine.newBuilder()
                .initialCapacity(100)
                .maximumSize(10000)
                // 1min后失效
                .expireAfterWrite(1, TimeUnit.MINUTES)
                .recordStats().build()));
		
        caches.add(new CaffeineCache(CacheConstants.GUAVA_1H_CACHE_SAFE, Caffeine.newBuilder()
                .initialCapacity(100)
                .maximumSize(10000)
                // 1h后失效
                .expireAfterWrite(1, TimeUnit.HOURS)
                .recordStats().build()));

        simpleCacheManager.setCaches(caches);
        return simpleCacheManager;
    }

    @Override
    public CacheResolver cacheResolver() {
        return null;
    }


    @Bean
    @Override
    // 生成缓存中检索的键值
    public KeyGenerator keyGenerator() {
    	// 使用方法对象、方法名称、方法参数进行区分
        return (target, method, params) -> {
            // 参数序列化,拼接类名 方法名
            List<String> args = Lists.newArrayList();
            if (params != null) {
                args = Arrays.stream(params).map(Object::toString).collect(Collectors.toList());
            }
            return Joiner.on("_").join(Lists.newArrayList(target.getClass().getSimpleName(),
                    method.getName(), StringUtils.arrayToDelimitedString(args.toArray(), "_")));
        };

    }

    @Override
    public CacheErrorHandler errorHandler() {
        return null;
    }
}

使用实例:
第一次调用时会走方法内部,之后就以键值对的形式存储在cache中。只要传入的入参凑成的键能在cache中查到,就会取缓存中的数据。1h后失效会再走方法生成。

@Cacheable(cacheNames = CacheConstants.GUAVA_1H_CACHE_SAFE)
    public String getAccessToken(String appId, String appSecret) {
        // token生成逻辑
        return larkToken;
    }

注意点:
1,持有cache的引用修改了字段后会同步修改cache中的字段,需要考虑线程安全性。
2,@Cacheable底层使用AOP实现,需要保证调用该方法时在Spring框架下调用的,否则将无法进入AOP。比如service自调用的情况就会导致缓存失效。

你可能感兴趣的:(Spring)