Java Spring-Cache key配置注意事項(xiàng)介紹
為了提升項(xiàng)目的并發(fā)性能,考慮引入本地內(nèi)存Cache,對(duì):外部數(shù)據(jù)源訪問(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實(shí)現(xiàn)優(yōu)雅日期處理的方案詳解
在我們的日常工作中,需要經(jīng)常處理各種格式,各種類(lèi)似的的日期或者時(shí)間,下面我們就來(lái)看看如何使用java處理這樣的日期問(wèn)題吧,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-04-04
Spring3 整合MyBatis3 配置多數(shù)據(jù)源動(dòng)態(tài)選擇SqlSessionFactory詳細(xì)教程
這篇文章主要介紹了Spring3 整合MyBatis3 配置多數(shù)據(jù)源動(dòng)態(tài)選擇SqlSessionFactory詳細(xì)教程,需要的朋友可以參考下2017-04-04
Java for-each循環(huán)使用難題2例(高級(jí)使用方法)
從Java5起,在Java中有了for-each循環(huán),可以用來(lái)循環(huán)遍歷collection和array。For each循環(huán)允許你在無(wú)需保持傳統(tǒng)for循環(huán)中的索引,或在使用iterator /ListIterator時(shí)無(wú)需調(diào)用while循環(huán)中的hasNext()方法就能遍歷collection2014-04-04
Java SpringBoot自動(dòng)裝配原理詳解及源碼注釋
SpringBoot的自動(dòng)裝配是拆箱即用的基礎(chǔ),也是微服務(wù)化的前提。其實(shí)它并不那么神秘,我在這之前已經(jīng)寫(xiě)過(guò)最基本的實(shí)現(xiàn)了,大家可以參考這篇文章,來(lái)看看它是怎么樣實(shí)現(xiàn)的,我們透過(guò)源代碼來(lái)把握自動(dòng)裝配的來(lái)龍去脈2021-10-10
Java并發(fā)編程——volatile關(guān)鍵字
這篇文章主要介紹了Java并發(fā)編程——volatile關(guān)鍵字的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)Java并發(fā)編程,感興趣的朋友可以了解下2020-10-10

