参考
底层实现: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自调用的情况就会导致缓存失效。