Java Spring-Cache key配置注意事項介紹
為了提升項目的并發(fā)性能,考慮引入本地內存Cache,對:外部數據源訪問、Restful API調用、可重用的復雜計算 等3種類型的函數處理結果進行緩存。目前采用的是spring Cache的@Cacheable注解方式,緩存具體實現選取的是Guava Cache。
具體緩存的配置此處不再介紹,重點對于key的配置進行說明:
1、基本形式
@Cacheable(value="cacheName", key"#id") public ResultDTO method(int id);
2、組合形式
@Cacheable(value="cacheName", key"T(String).valueOf(#name).concat('-').concat(#password)) public ResultDTO method(int name, String password);
3、對象形式
@Cacheable(value="cacheName", key"#user.id) public ResultDTO method(User user);
4、自定義key生成器
@Cacheable(value="gomeo2oCache", keyGenerator = "keyGenerator") public ResultDTO method(User user);
注意:Spring默認的SimpleKeyGenerator是不會將函數名組合進key中的
如下:
@Component public class CacheTestImpl implements CacheTest { @Cacheable("databaseCache") public Long test1() { return 1L; } @Cacheable("databaseCache") public Long test2() { return 2L; } @Cacheable("databaseCache") public Long test3() { return 3L; } @Cacheable("databaseCache") public String test4() { return "4"; } }
我們期望輸出:
1 2 3 4
實際卻輸出:
1 1 1 ClassCastException: java.lang.Long cannot be cast to java.lang.String
此外,原子類型的數組,直接作為key使用也是不會生效的
為了解決上述2個問題,自定義了一個KeyGenerator如下:
class CacheKeyGenerator implements KeyGenerator { // custom cache key public static final int NO_PARAM_KEY = 0; public static final int NULL_PARAM_KEY = 53; @Override public Object generate(Object target, Method method, Object... params) { StringBuilder key = new StringBuilder(); key.append(target.getClass().getSimpleName()).append(".").append(method.getName()).append(":"); if (params.length == 0) { return key.append(NO_PARAM_KEY).toString(); } for (Object param : params) { if (param == null) { log.warn("input null param for Spring cache, use default key={}", NULL_PARAM_KEY); key.append(NULL_PARAM_KEY); } else if (ClassUtils.isPrimitiveArray(param.getClass())) { int length = Array.getLength(param); for (int i = 0; i < length; i++) { key.append(Array.get(param, i)); key.append(','); } } else if (ClassUtils.isPrimitiveOrWrapper(param.getClass()) || param instanceof String) { key.append(param); } else { log.warn("Using an object as a cache key may lead to unexpected results. " + "Either use @Cacheable(key=..) or implement CacheKey. Method is " + target.getClass() + "#" + method.getName()); key.append(param.hashCode()); } key.append('-'); } String finalKey = key.toString(); long cacheKeyHash = Hashing.murmur3_128().hashString(finalKey, Charset.defaultCharset()).asLong(); log.debug("using cache key={} hashCode={}", finalKey, cacheKeyHash); return key.toString(); } }
采用此方式后可以解決:多參數、原子類型數組、方法名識別 等問題
總結
以上就是本文關于Java Spring-Cache key配置注意事項介紹的全部內容,感興趣的朋友可以繼續(xù)參閱:spark之Standalone模式部署配置詳解、struts2開發(fā)流程及詳細配置、Java之Spring注解配置bean實例代碼解析等,如有不足之處,歡迎留言指出,小編會及時回復大家并修正,給廣大編程愛好者提供更好的閱讀體驗,希望對大家有所幫助。在此也非常希望朋友們對本站多多支持!
相關文章
JAVA內存模型和Happens-Before規(guī)則知識點講解
在本篇文章里小編給大家整理的是一篇關于JAVA內存模型和Happens-Before規(guī)則知識點內容,有需要的朋友們跟著學習下。2020-11-11SpringBoot中整合MyBatis-Plus的方法示例
這篇文章主要介紹了SpringBoot中整合MyBatis-Plus的方法示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-09-09java使用listIterator逆序arraylist示例分享
對于列表而言,除了Iterator,還提供了一個功能更加強大的ListIterator。它可以實現逆序遍歷列表中的元素。本示例將使用其逆序遍歷ArrayList2014-02-02Java中BeanUtils.copyProperties()詳解及應用場景
BeanUtils.copyProperties()是Apache?Commons?BeanUtils提供的方法,用于Java對象間屬性的復制,特別適用于DTO、VO和Entity之間的數據傳遞,這篇文章主要介紹了Java中BeanUtils.copyProperties()詳解及應用場景的相關資料,需要的朋友可以參考下2024-09-09