亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

spring緩存自定義resolver的方法

 更新時間:2022年03月14日 14:30:52   作者:QiHY  
這篇文章主要為大家詳細介紹了spring緩存自定義resolver的方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助

本文介紹spring中自定義緩存resolver,通過自定義resolver,可以在spring的cache注解中增加附加處理。

一、概述

cache-aside模式是常用的緩存使用模式。

使用流程如下圖:

在這里插入圖片描述

當更新數(shù)據庫中的數(shù)據后,對緩存做失效處理,后續(xù)就能讀取到數(shù)據庫中最新的數(shù)據,使得緩存數(shù)據與數(shù)據庫數(shù)據保持一致。

在spring中通過cache注解進行緩存的處理,一般會把緩存處理封裝到dao層,這樣業(yè)務層就不需要感知緩存操作的細節(jié),可以專注于業(yè)務邏輯的處理。

二、緩存的讀取和失效

dao層的操作通常使用springdatajpa,數(shù)據庫方法都是一個interface,通過在interface上增加對應的cache注解實現(xiàn)緩存處理。

讀取數(shù)據:

@Cacheable(value = "testCache", key = "#p0", unless = "#result == null")
Optional<DemoEntity> findById(Long id);

通過Cacheable注解,從數(shù)據庫中讀取到數(shù)據后,會同步寫到緩存中。

保存數(shù)據:

@CacheEvict(value = "testCache", key = "#p0.id")
DemoEntity save(DemoEntity entity);

通過CacheEvict注解,在將數(shù)據寫入到數(shù)據庫后,對緩存進行失效。

如果我們想在緩存失效后,進行其它的操作,例如將失效緩存的key寫入kafka,用于其它系統(tǒng)同步刪除緩存,這時該怎樣處理?

三、自定義緩存resolver

spring提供了自定義緩存resolver的方式,通過自定義resolver,可以在緩存處理中增加附加操作。

@Configuration
public class RedisCacheConfig extends CachingConfigurerSupport {
    @Bean
    public RedisCacheManager redisCacheManager(RedisConnectionFactory redisConnectionFactory) {
        RedisCacheConfiguration cacheConfiguration = RedisCacheConfiguration.defaultCacheConfig()
                .computePrefixWith(cacheName -> cacheName.concat(":"));
        return RedisCacheManager.builder(redisConnectionFactory)
                .cacheDefaults(cacheConfiguration)
                .build();
    }
    @Bean
    public CacheResolver customCacheResolver(RedisConnectionFactory redisConnectionFactory) {
        return new CustomCacheResolver(redisCacheManager(redisConnectionFactory));
    }
}

以上代碼是redis緩存的配置,其中 RedisCacheManager部分是常規(guī)的cacheManager的配置, 而 customCacheResolver部分是自定義resolver的配置,通過定義customCacheResolver這個bean,可以在cache注解中引用到這個自定義的resolver。

定義好customCacheResolver的bean后,我們就可以在cache注解中引用,上面提到的數(shù)據保存方法改造后的代碼:

@CacheEvict(value = "testCache", cacheResolver = "customCacheResolver", key = "#p0.id")
DemoEntity save(DemoEntity entity);

相比于之前的實現(xiàn),對CacheEvict增加指定cacheResolver。

四、自定義resolver的實現(xiàn)

上面我們介紹了如果配置和引用cacheResolver,下面介紹自定義cacheResolver的實現(xiàn)。

public class CustomCacheResolver extends SimpleCacheResolver {
    public CustomCacheResolver(CacheManager cacheManager) {
        super(cacheManager);
    }
    @Override
    @NonNull
    public Collection<? extends Cache> resolveCaches(CacheOperationInvocationContext<?> context) {
        ParameterNameDiscoverer paramNameDiscoverer = new DefaultParameterNameDiscoverer();
        EvaluationContext evaluationContext = new MethodBasedEvaluationContext(context.getOperation(), context.getMethod(), context.getArgs(), paramNameDiscoverer);
        Expression exp = (new SpelExpressionParser()).parseExpression(((CacheEvictOperation) context.getOperation()).getKey());
        Collection<? extends Cache> caches = super.resolveCaches(context);
        context.getOperation().getCacheNames().forEach(cacheName -> {
            String key = cacheName + ':' + exp.getValue(evaluationContext, String.class);
            log.info("cache key={}", key);
        });
        return caches;
    }
}

上面的代碼定義了CustomCacheResolver這個自定義resolver類,繼承SimpleCacheResolverSimpleCacheResolver類是spring在cache注解中默認使用的resolver。

我們通過擴展SimpleCacheResolver這個類,來增加附加操作。其中resolveCaches就是解析緩存操作的部分。

在這部分代碼中,我們需要的是獲取到 @CacheEvict(value = "testCache", cacheResolver = "customCacheResolver", key = "#p0.id")注解中失效的緩存的key的值。

通過 context.getOperation()).getKey()從參數(shù)context中可以讀取到key的定義,即 #p0.id,這個定義是一個spel表達式,與普通的spel表達式不同, p0這個變量是jpa方法中的一個特有變量,表示方法中的第一個參數(shù),同樣p1表示方法中的第二個參數(shù)。通過普通的spel處理無法解析這個spel表達式。
spring提供了 MethodBasedEvaluationContext類用于解析這種特殊的spel表達式。

通過一下四行代碼,我們就能夠獲取到具體的key的值:

ParameterNameDiscoverer paramNameDiscoverer = new DefaultParameterNameDiscoverer();
EvaluationContext evaluationContext = new MethodBasedEvaluationContext(context.getOperation(), context.getMethod(), context.getArgs(), paramNameDiscoverer);
Expression exp = (new SpelExpressionParser()).parseExpression(((CacheEvictOperation) context.getOperation()).getKey());
String key = cacheName + ':' + exp.getValue(evaluationContext, String.class);

獲取到了key的值,我們就可以對這個key做很多操作,可以把這個key寫入kafka,通知其它系統(tǒng)同步清理key。

五、總結

我們通常把緩存操作封裝到dao層以簡化程序的整體邏輯,當使用springdatajpa作為dao層的實現(xiàn)時,具體的dao方法都是interface,對于在interface上添加的cache注解,沒有辦法增加額外的其它操作。

當需要對緩存操作做額外處理時,可以通過自定義resolver的方式實現(xiàn),在cache注解中使用我們自定義的resolver。

這樣既沒有破環(huán)程序的整理邏輯,又擴展了對緩存的操作,是一種比較好的實現(xiàn)方式。

本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關注腳本之家的更多內容!  

相關文章

  • Spring?Boot提高開發(fā)效率必備工具lombok使用

    Spring?Boot提高開發(fā)效率必備工具lombok使用

    這篇文章主要為大家介紹了Spring?Boot提高開發(fā)效率的必備工具lombok使用方法示例及步驟說明,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-03-03
  • Java中Date日期時間類具體使用

    Java中Date日期時間類具體使用

    本文主要介紹了Java中Date日期時間類具體使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-04-04
  • Java經典快排思想以及快排的改進講解

    Java經典快排思想以及快排的改進講解

    今天小編就為大家分享一篇關于Java經典快排思想以及快排的改進講解,小編覺得內容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-01-01
  • springboot中的css樣式顯示不出了的幾種情況

    springboot中的css樣式顯示不出了的幾種情況

    這篇文章主要介紹了springboot中的css樣式顯示不出了的幾種情況,具有很好的的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • Java中Timer的schedule()方法參數(shù)詳解

    Java中Timer的schedule()方法參數(shù)詳解

    今天小編就為大家分享一篇關于Java中Timer的schedule()方法參數(shù)詳解,小編覺得內容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-03-03
  • java基礎中異常及包歸納整理

    java基礎中異常及包歸納整理

    這篇文章主要介紹了java基礎中的 異常與包,有需要的朋友可以參考一下
    2017-04-04
  • java項目實現(xiàn)統(tǒng)一打印入參出參等日志

    java項目實現(xiàn)統(tǒng)一打印入參出參等日志

    這篇文章主要介紹了java項目實現(xiàn)統(tǒng)一打印入參出參等日志方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • SpringBoot集成Redisson實現(xiàn)分布式鎖的方法示例

    SpringBoot集成Redisson實現(xiàn)分布式鎖的方法示例

    這篇文章主要介紹了SpringBoot集成Redisson實現(xiàn)分布式鎖的方法示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-10-10
  • 使用SpringCache操作Redis緩存數(shù)據的示例代碼

    使用SpringCache操作Redis緩存數(shù)據的示例代碼

    SpringCache是一個框架,實現(xiàn)了基于注解的緩存功能,只需要簡單的加一個注解,就能實現(xiàn)緩存功能,本文給大家介紹了如何使用SpringCache操作Redis緩存數(shù)據,文中有相關的代碼示例供大家參考,需要的朋友可以參考下
    2024-01-01
  • swagger2和knife4j的詳細使用教程(入門級)

    swagger2和knife4j的詳細使用教程(入門級)

    最近項目中用到了Swagger2和knife4j作為接口文檔,所以下面這篇文章主要給大家介紹了關于swagger2和knife4j的詳細使用教程,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-09-09

最新評論