Java Spring-Cache key配置注意事項(xiàng)介紹
為了提升項(xiàng)目的并發(fā)性能,考慮引入本地內(nèi)存Cache,對(duì):外部數(shù)據(jù)源訪(fǎng)問(wèn)、Restful API調(diào)用、可重用的復(fù)雜計(jì)算 等3種類(lèi)型的函數(shù)處理結(jié)果進(jìn)行緩存。目前采用的是spring Cache的@Cacheable注解方式,緩存具體實(shí)現(xiàn)選取的是Guava Cache。
具體緩存的配置此處不再介紹,重點(diǎn)對(duì)于key的配置進(jìn)行說(shuō)明:
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、對(duì)象形式
@Cacheable(value="cacheName", key"#user.id) public ResultDTO method(User user);
4、自定義key生成器
@Cacheable(value="gomeo2oCache", keyGenerator = "keyGenerator") public ResultDTO method(User user);
注意:Spring默認(rèn)的SimpleKeyGenerator是不會(huì)將函數(shù)名組合進(jìn)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
實(shí)際卻輸出:
1 1 1 ClassCastException: java.lang.Long cannot be cast to java.lang.String
此外,原子類(lèi)型的數(shù)組,直接作為key使用也是不會(huì)生效的
為了解決上述2個(gè)問(wèn)題,自定義了一個(gè)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(); } }
采用此方式后可以解決:多參數(shù)、原子類(lèi)型數(shù)組、方法名識(shí)別 等問(wèn)題
總結(jié)
以上就是本文關(guān)于Java Spring-Cache key配置注意事項(xiàng)介紹的全部?jī)?nèi)容,感興趣的朋友可以繼續(xù)參閱:spark之Standalone模式部署配置詳解、struts2開(kāi)發(fā)流程及詳細(xì)配置、Java之Spring注解配置bean實(shí)例代碼解析等,如有不足之處,歡迎留言指出,小編會(huì)及時(shí)回復(fù)大家并修正,給廣大編程愛(ài)好者提供更好的閱讀體驗(yàn),希望對(duì)大家有所幫助。在此也非常希望朋友們對(duì)本站多多支持!
相關(guān)文章
JAVA內(nèi)存模型和Happens-Before規(guī)則知識(shí)點(diǎn)講解
在本篇文章里小編給大家整理的是一篇關(guān)于JAVA內(nèi)存模型和Happens-Before規(guī)則知識(shí)點(diǎn)內(nèi)容,有需要的朋友們跟著學(xué)習(xí)下。2020-11-11SpringBoot中整合MyBatis-Plus的方法示例
這篇文章主要介紹了SpringBoot中整合MyBatis-Plus的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09java使用listIterator逆序arraylist示例分享
對(duì)于列表而言,除了Iterator,還提供了一個(gè)功能更加強(qiáng)大的ListIterator。它可以實(shí)現(xiàn)逆序遍歷列表中的元素。本示例將使用其逆序遍歷ArrayList2014-02-02Java中BeanUtils.copyProperties()詳解及應(yīng)用場(chǎng)景
BeanUtils.copyProperties()是Apache?Commons?BeanUtils提供的方法,用于Java對(duì)象間屬性的復(fù)制,特別適用于DTO、VO和Entity之間的數(shù)據(jù)傳遞,這篇文章主要介紹了Java中BeanUtils.copyProperties()詳解及應(yīng)用場(chǎng)景的相關(guān)資料,需要的朋友可以參考下2024-09-09Java中如何使用正則表達(dá)式提取各種類(lèi)型括號(hào)中的內(nèi)容
最近在工作中遇到一個(gè)問(wèn)題,就是需要一個(gè)字符串中每一個(gè)中括號(hào)里的內(nèi)容,下面這篇文章主要給大家介紹了關(guān)于Java中如何使用正則表達(dá)式提取各種類(lèi)型括號(hào)中的內(nèi)容,需要的朋友可以參考下2023-06-06MyBatis關(guān)聯(lián)查詢(xún)的實(shí)現(xiàn)
MyBatis可以通過(guò)定義多個(gè)表的關(guān)聯(lián)關(guān)系,實(shí)現(xiàn)多表查詢(xún),本文主要介紹了MyBatis關(guān)聯(lián)查詢(xún)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2023-11-11Java和SQL實(shí)現(xiàn)取兩個(gè)字符間的值
這篇文章主要介紹了Java和SQL實(shí)現(xiàn)取兩個(gè)字符間的值操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06