91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么使用caffeine_redis自定義二級緩存

發布時間:2023-04-19 14:25:38 來源:億速云 閱讀:144 作者:iii 欄目:開發技術

這篇文章主要介紹了怎么使用caffeine_redis自定義二級緩存的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇怎么使用caffeine_redis自定義二級緩存文章都會有所收獲,下面我們一起來看看吧。

    問題

    基于提出的需求,我認為主要有以下兩個問題:

    • 因為有本地緩存,如何保證數據一致性。當一個節點數據改變,其他節點的數據如何失效?

    • 數據不對,需要重新同步,緩存如何失效?

    流程圖

    接下來就是配合產品和其他開發人員畫出流程圖,如下:

    • 使用一張配置表,記錄是否需要緩存,是否開啟緩存,來達到通知時候緩存失效的情況。

    • 因為項目要求一般,即使消息丟失,也不會存在太大的影響,所以最終選擇了 redis 里面的訂閱、發布功能,實現通知其他節點失效本地緩存。

    開發

    上面問題清楚了,流程圖也清楚了。那就準備開始寫 bug 了。整體思路是自定義注解實現切面,盡量降低對業務代碼的耦合度。

    CacheConfig

    主要是結合業務定義一個 CacheManager,代碼里面的解釋都有。因為這個是直接占用程序內存的,所有得特別注意最大可緩存條數,別把內存肝爆了。當然也不能太小了,因為還要考慮命中率的問題。所以這就得結合實際得業務來確定最終的大小。

    @Bean(name = JKDDCX)
    @Primary
    public CacheManager cacheManager() {
         CaffeineCacheManager cacheManager  = new CaffeineCacheManager();
            cacheManager.setCaffeine(Caffeine.newBuilder()
                    // 設置最后一次寫入或訪問后經過固定時間過期
                    .expireAfterAccess(EXPIRE, TIME_UNIT)
                    //設置本地緩存寫入后過期時間
                    .expireAfterWrite(EXPIRE, TIME_UNIT)
                    // 初始的緩存空間大小
                    .initialCapacity(500)
                    // 緩存的最大條數
                    .maximumSize(1000));// 使用人數 * 5 (每個人不同的入參 5 條)\
      return cacheManager;
    }

    @CaffeineCache

    自定義注解,把可以用到的參數都能加上。

    @Target({ ElementType.METHOD ,ElementType.TYPE})
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface CaffeineCache {
    
         public String moudleId() default "";
        
         //用于在數據庫中配置參數
         public String methodId() default "";
         public String cachaName() default "";
        
         //動態切換實際的 CacheManager
         public String cacheManager() default "";
    
    }

    CacheMessageListener

    緩存監聽器,主要是保證多節點數據一致性的問題。當一個節點緩存更新,通知其他的節點相應處理。主要技術是 Redis 的發布、訂閱功能,實現 MessageListener 接口。

    當然下面還有個細節就是一般生產環境是禁用 Redis#keys 命令的,所以得換個方式掃描對應的 key。

    public class CacheMessageListener implements MessageListener {
         @Override
        public void onMessage(Message message, byte[] pattern) {
            CacheMessage cacheMessage = (CacheMessage) redisTemplate.getValueSerializer().deserialize(message.getBody());
            logger.info("收到redis清除緩存消息, 開始清除本地緩存, the cacheName is {}, the key is {}", cacheMessage.getCacheName(), cacheMessage.getKey());
    //		redisCaffeineCacheManager.clearLocal(cacheMessage.getCacheName(), cacheMessage.getKey());
    
            /**
             * 如果是一個類上使用了 注解 @CaffeineCache ,那么所有接口都會緩存。
             * 下面的邏輯是:除了當前模塊的接口訪問的入參 key,其他的 redis 緩存都會被清除
             * (比如此模塊的表更新了,但是當前調用此接口只是緩存了當前這個入參的redis,其他的數據刪除)
             */
            String prefixKey = RedisConstant.WXYMG_DATA_CACHE + cacheMessage.getCacheName();
            Set<String> keys = redisTemplate.execute((RedisCallback<Set<String>>) connection -> {
                Set<String> keysTmp = new HashSet<>();
                Cursor<byte[]> cursor = connection.scan(new ScanOptions.ScanOptionsBuilder().
                        match(prefixKey + "*").
                        count(50).build());
                while (cursor.hasNext()) {
                    keysTmp.add(new String(cursor.next()));
                }
                return keysTmp;
            });
            Iterator iterator = keys.iterator();
            while (iterator.hasNext()) {
                if (iterator.next().toString().equals(cacheMessage.getKey())) {
                    iterator.remove();
                }
            }
            redisTemplate.delete(keys);
    
            cacheConfig.cacheManager().getCache(cacheMessage.getCacheName()).clear(); //cacheName 下的都刪除
        }
    }

    CaffeineCacheAspect

    然后就是切面的邏輯處理,里面的內容和 流程圖 一模一樣,只是使用代碼實現了需求。

    其中:下面的代碼是 Redis 發布消息。

    redisTemplate.convertAndSend(CacheConfig.TOPIC, new CacheMessage(caffeineCache.cachaName(), redisKey));

    CacheMessage

    這是在 Redis 發布消息的時候一個消息體,也是自定義的,可以加更多的參數屬性

    public class CacheMessage implements Serializable {
    
    	private static final long serialVersionUID = -1L;
    
    	private String cacheName;
    
    	private Object key;
    
    	public CacheMessage(String cacheName, Object key) {
    		super();
    		this.cacheName = cacheName;
    		this.key = key;
    	}
    
    }

    關于“怎么使用caffeine_redis自定義二級緩存”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“怎么使用caffeine_redis自定義二級緩存”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注億速云行業資訊頻道。

    向AI問一下細節

    免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

    AI

    商河县| 曲阜市| 丰都县| 大石桥市| 宽甸| 金堂县| 荥阳市| 阿瓦提县| 平舆县| 竹北市| 婺源县| 安康市| 保德县| 太和县| 安陆市| 蓝山县| 禄丰县| 南岸区| 东兴市| 秭归县| 林口县| 黄浦区| 保定市| 屏东县| 防城港市| 黄石市| 青冈县| 石柱| 长寿区| 襄汾县| 马关县| 北宁市| 隆安县| 灵武市| 龙海市| 马龙县| 丽水市| 贵港市| 金秀| 土默特右旗| 莱州市|