ehcache模糊批量移除緩存的方法
前言
眾所周知,encache是現(xiàn)在最流行的java開(kāi)源緩存框架,配置簡(jiǎn)單,結(jié)構(gòu)清晰,功能強(qiáng)大。通過(guò)注解 @Cacheable 可以快速添加方法結(jié)果到緩存。通過(guò) @CacheEvict 可以快速清除掉指定的緩存。
但由于 @CacheEvict 注解使用的是key-value的,不支持模糊刪除,就會(huì)遇到問(wèn)題。當(dāng)我用 @Cacheable 配合Spring EL表達(dá)式添加了同一方法的多個(gè)緩存比如:
@GetMapping("/listOfTask/{page}/")
@Cacheable(value = "BusinessCache", key = "'listOfTask_'+ #page")
public ResponseMessage<PageTaskVO> getTaskList(@PathVariable("page") String page) {
do something...
}
上述代碼是分頁(yè)獲取任務(wù)信息。用EL表達(dá)式獲取到參數(shù)中的page,并作為緩存的key,使用 @Cacheable 添加到ehcache的緩存中。此時(shí),在緩存中就會(huì)出現(xiàn) listOfTask_1 , listOfTask_2 , listOfTask_3 這種類型的key。
當(dāng)添加、刪除任務(wù)時(shí),列表就會(huì)發(fā)生改變。這時(shí)候,就需要把 listOfTask_* 相關(guān)的緩存全部去掉。而這時(shí),我不知道緩存中到底緩存了多少和 listOfTask_* 相關(guān)的內(nèi)容,不可能調(diào)用 @CacheEvict 挨個(gè)刪除。
既然ehcache本身無(wú)法支持,那就只能靠我們自己實(shí)現(xiàn)了。
實(shí)現(xiàn)
考慮到使用的注解添加的緩存,那么移除緩存也使用注解處理,可以保持開(kāi)發(fā)的一致性。注解對(duì)開(kāi)發(fā)者來(lái)說(shuō)也很友好。那么我們就考慮使用自定義注解來(lái)來(lái)模糊批量移除緩存。
首先,定義注解 CacheRemove :
@Target({ java.lang.annotation.ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface CacheRemove {
String value();
String[] key();
}
其中,value 同 ehcache 一樣,用于定義要操作的緩存名。key 是一個(gè)數(shù)組,用于存放多種緩存 key 的正則表達(dá)式。起名 CacheRemove 清晰易懂,也不與 ehcache 本身的注解沖突。注解的定義到此為止。接下來(lái),就需要處理注解了,由于使用的 spring 框架,很自然的,就會(huì)想到用 AOP 來(lái)做注解的具體實(shí)現(xiàn)。
注解的目的是批量模糊移除緩存。需考慮如下兩個(gè)問(wèn)題:
- 用什么方式模糊匹配
- 怎么批量刪除key
我給出的處理方式,也是我認(rèn)為最簡(jiǎn)單的處理方式是:
- 用什么方式模糊匹配 —— CacheRemove 中的key傳正則,可以傳多個(gè),使用正則匹配
- 怎么批量刪除key —— 循環(huán)所有的key,找到匹配正則的就刪除
首先定義類名 CacheRemoveAspect :
@Aspect
@Component
public class CacheRemoveAspect {
@Pointcut(value = "(execution(* *.*(..)) && @annotation(com.example.CacheRemove))")
private void pointcut() {}
do something...
}
在切面中定義切點(diǎn),使用 execution(* *.*(..) && @annotation(com.example.CacheRemove)) 表示所有帶注解類 CacheRemove 都執(zhí)行, @annotation 中的值是注解的全限定名。
切點(diǎn)定義完畢,下面的重頭戲就是切面的具體實(shí)現(xiàn)了。一般來(lái)說(shuō),緩存會(huì)在增刪改的方法執(zhí)行完后才要移除。所以使用 @AfterReturning() 來(lái)實(shí)現(xiàn)。在具體實(shí)現(xiàn)中需要做以下幾件事:
- 攔截方法上的注解
- 判斷注解是不是 CacheRemove
- 由于注解傳入的 key 是個(gè)數(shù)組,循環(huán)處理每個(gè)key
- 在循環(huán)中編制每個(gè) key 為 pattern, 并循環(huán)所有的緩存,移除匹配上的緩存
具體實(shí)現(xiàn)如下:
@AfterReturning(value = "pointcut()")
private void process(JoinPoint joinPoint){
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
Method method = signature.getMethod();
CacheRemove cacheRemove = method.getAnnotation(CacheRemove.class);
if (cacheRemove != null){
String value = cacheRemove.value();
String[] keys = cacheRemove.key(); //需要移除的正則key
List cacheKeys = CacheUtils.cacheKeys(value);
for (String key : keys){
Pattern pattern = Pattern.compile(key);
for (Object cacheKey: cacheKeys) {
String cacheKeyStr = String.valueOf(cacheKey);
if (pattern.matcher(cacheKeyStr).find()){
CacheUtils.remove(value, cacheKeyStr);
}
}
}
}
}
以上,為 ehcache 模糊批量移除緩存的具體實(shí)現(xiàn)。其中 BusinessCacheUtils 為自己封裝的 ehcache 工具類。主要實(shí)現(xiàn)獲取緩存池,獲取緩存,移除緩存,添加緩存,查看所有緩存等正常功能。代碼如下:
public class CacheUtils {
private static CacheManager cacheManager = SpringContextHolder.getBean("ehCacheManagerFactory");
public static Object get(String cacheName, String key) {
Element element = getCache(cacheName).get(key);
return element == null ? null : element.getObjectValue();
}
public static void put(String cacheName, String key, Object value) {
Element element = new Element(key, value);
getCache(cacheName).put(element);
}
public static void remove(String cacheName, String key) {
getCache(cacheName).remove(key);
}
public static List cacheKeys(String cacheName){
return getCache(cacheName).getKeys();
}
/**
* 獲得一個(gè)Cache,沒(méi)有則創(chuàng)建一個(gè)。
* @param cacheName
* @return
*/
private static Cache getCache(String cacheName) {
Cache cache = cacheManager.getCache(cacheName);
if (cache == null) {
cacheManager.addCache(cacheName);
cache = cacheManager.getCache(cacheName);
cache.getCacheConfiguration().setEternal(true);
}
return cache;
}
public static CacheManager getCacheManager() {
return cacheManager;
}
}
至此,整個(gè)ehcache 模糊批量移除緩存的功能就實(shí)現(xiàn)了。
總結(jié)
整個(gè)過(guò)程思路簡(jiǎn)單,用到了一些 AOP 的知識(shí)就完成了需要的功能。但具體的移除部分代碼可考慮進(jìn)行優(yōu)化。通過(guò)一次緩存的全部循環(huán),就把需要移除的緩存都移除干凈,而不是想現(xiàn)在這樣有幾個(gè)key,就全緩存遍歷幾次。具體實(shí)現(xiàn)留給讀者自行完成。希望對(duì)各位有所幫助。也希望大家多多支持腳本之家。
相關(guān)文章
Spring Boot使用線程池創(chuàng)建多線程的完整示例
在 Spring Boot 2 中,可以使用 @Autowired 注入 線程池(ThreadPoolTaskExecutor 或 ExecutorService),從而管理線程的創(chuàng)建和執(zhí)行,以下是使用 @Autowired 方式注入線程池的完整示例,感興趣的朋友一起看看吧2025-03-03
詳解全局事務(wù)注解@GlobalTransactional的識(shí)別
這篇文章主要為大家介紹了詳解全局事務(wù)注解@GlobalTransactional的識(shí)別源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12
java Class文件結(jié)構(gòu)解析常量池字節(jié)碼
這篇文章主要為大家介紹了java Class文件的整體結(jié)構(gòu)解析常量池字節(jié)碼詳細(xì)講解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
相冊(cè)管理系統(tǒng)(Java表單+xml數(shù)據(jù)庫(kù)存儲(chǔ))
這篇文章主要為大家詳細(xì)介紹了相冊(cè)管理系統(tǒng)的實(shí)現(xiàn)步驟,Java表單的文件上傳和下載,xml數(shù)據(jù)庫(kù)存儲(chǔ)信息,感興趣的小伙伴們可以參考一下2016-07-07
vue+ java 實(shí)現(xiàn)多級(jí)菜單遞歸效果
這篇文章主要介紹了vue+ java 實(shí)現(xiàn)多級(jí)菜單遞歸效果,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-12-12
MybatisPlus多條件?or()的使用問(wèn)題小結(jié)
這篇文章主要介紹了MybatisPlus多條件?or()的使用問(wèn)題小結(jié),本文給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-05-05
解決使用this.getClass().getResource()獲取文件時(shí)遇到的坑
這篇文章主要介紹了解決使用this.getClass().getResource()獲取文件時(shí)遇到的坑問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12

