redisson滑動(dòng)時(shí)間窗應(yīng)用場(chǎng)景解決方案
概述
前10分鐘內(nèi)累計(jì)3次驗(yàn)證失敗后,增加圖形驗(yàn)證碼驗(yàn)證條件,前10分鐘內(nèi)累計(jì)6次驗(yàn)證失敗后,系統(tǒng)自動(dòng)鎖定該賬號(hào)15分鐘,15分鐘后自動(dòng)解鎖;
方案
基于redisson(zset)滑動(dòng)時(shí)間窗記錄最近10分鐘內(nèi)該賬戶登錄失敗次數(shù)
統(tǒng)計(jì)次數(shù)、每次失敗加1
/** * 統(tǒng)計(jì)請(qǐng)求次數(shù) * * @param windowTime 窗口時(shí)間,單位:秒 * @param key 登錄賬戶 * @return 請(qǐng)求次數(shù) */ public Integer count(int windowTime, String key) { // 窗口結(jié)束時(shí)間 long windowEndTime = System.currentTimeMillis(); // 窗口開(kāi)始時(shí)間 long windowStartTime = windowEndTime - windowTime * 1000L; try { // 創(chuàng)建 RBatch 實(shí)例,批量執(zhí)行命令 RBatch batch = redissonClient.createBatch(); // 添加元素 score=當(dāng)前時(shí)間戳 value=請(qǐng)求序列號(hào),唯一不可重復(fù) batch.getScoredSortedSet(key).addAsync(windowEndTime, UUID.randomUUID().toString()); // 統(tǒng)計(jì)數(shù)據(jù) batch.getScoredSortedSet(key).countAsync(windowStartTime, true, windowEndTime, true); // 清除窗口過(guò)期成員 batch.getScoredSortedSet(key).removeRangeByScoreAsync(0, true, windowStartTime, false); // 設(shè)置key過(guò)期時(shí)間 batch.getScoredSortedSet(key).expireAsync(Duration.ofSeconds(windowTime)); // 執(zhí)行管道命令 BatchResult<?> batchResult = batch.execute(); // 返回統(tǒng)計(jì)數(shù)量 List<?> responses = batchResult.getResponses(); Integer number = (Integer) responses.get(1); return number; } catch (Exception e) { log.error("統(tǒng)計(jì)請(qǐng)求次數(shù)異常!", e); return null; } }
獲取登錄失敗次數(shù)
/** * 獲取對(duì)應(yīng)窗口的次數(shù) * @param windowTime 窗口大小十分鐘,600s * @param key * @return */ public Integer getLastCachedCount(int windowTime, String key) { // 窗口結(jié)束時(shí)間 long windowEndTime = System.currentTimeMillis(); // 窗口開(kāi)始時(shí)間 long windowStartTime = windowEndTime - windowTime * 1000L; try { // 創(chuàng)建 RBatch 實(shí)例,批量執(zhí)行命令 RBatch batch = redissonClient.createBatch(); // 統(tǒng)計(jì)數(shù)據(jù),獲取上一次緩存存取的數(shù)據(jù) RFuture<Integer> countFuture = batch.getScoredSortedSet(key).countAsync(windowStartTime, true, windowEndTime, true); // 執(zhí)行管道命令 batch.execute(); // 等待統(tǒng)計(jì)數(shù)據(jù)完成并獲取結(jié)果 Integer count = countFuture.get(); // 如果結(jié)果為null,則返回0,否則返回計(jì)數(shù)值 return count == null ? 0 : count.intValue(); } catch (Exception e) { log.error("獲取上一次緩存存取數(shù)據(jù)異常!", e); // 如果出現(xiàn)異常,返回null return null; } }
清除登錄失敗次數(shù)
public void clearCachedCount(int windowTime, String key) { // 窗口結(jié)束時(shí)間 long windowEndTime = System.currentTimeMillis(); // 窗口開(kāi)始時(shí)間 long windowStartTime = windowEndTime - windowTime * 1000L; try { // 創(chuàng)建 RBatch 實(shí)例,批量執(zhí)行命令 RBatch batch = redissonClient.createBatch(); // 統(tǒng)計(jì)數(shù)據(jù),獲取上一次緩存存取的數(shù)據(jù) batch.getScoredSortedSet(key).removeRangeByScoreAsync(windowStartTime, true, windowEndTime, true); // 執(zhí)行管道命令 batch.execute(); // 如果結(jié)果為null,則返回0,否則返回計(jì)數(shù)值 } catch (Exception e) { log.error("清楚登錄失敗記錄次數(shù)異常", e); } }
到此這篇關(guān)于redisson滑動(dòng)時(shí)間窗應(yīng)用場(chǎng)景的文章就介紹到這了,更多相關(guān)redisson滑動(dòng)時(shí)間窗內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
RedisTemplate集成+封裝RedisUtil過(guò)程
本文介紹了如何搭建一個(gè)多模塊的Redis項(xiàng)目,包括項(xiàng)目搭建、配置和測(cè)試,通過(guò)使用父項(xiàng)目管理多個(gè)子模塊,可以實(shí)現(xiàn)單點(diǎn)構(gòu)建、統(tǒng)一版本管理和清晰的項(xiàng)目結(jié)構(gòu),文章還提供了在Spring Boot項(xiàng)目中集成RedisTemplate的示例,并解決了編碼問(wèn)題2024-12-12Redis如何使用樂(lè)觀鎖(CAS)保證數(shù)據(jù)一致性
本文主要介紹了Redis如何使用樂(lè)觀鎖(CAS)保證數(shù)據(jù)一致性,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03Redis序列化設(shè)置以及jetcache連接Redis序列化的設(shè)置過(guò)程
這篇文章主要介紹了Redis序列化設(shè)置以及jetcache連接Redis序列化的設(shè)置過(guò)程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12Redis數(shù)據(jù)庫(kù)的使用場(chǎng)景介紹(避免誤用Redis)
這篇文章主要介紹了Redis數(shù)據(jù)庫(kù)的使用場(chǎng)景介紹(避免誤用Redis),本文用簡(jiǎn)要的語(yǔ)言總結(jié)了Redis數(shù)據(jù)庫(kù)的適應(yīng)場(chǎng)合,人而避免錯(cuò)誤的使用它而產(chǎn)生昂貴的維護(hù)代價(jià),需要的朋友可以參考下2015-03-03使用 Redis 流實(shí)現(xiàn)消息隊(duì)列的代碼
這篇文章主要介紹了使用 Redis 流實(shí)現(xiàn)消息隊(duì)列,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-11-11Redis基于Bitmap實(shí)現(xiàn)用戶簽到功能
很多應(yīng)用上都有用戶簽到的功能,尤其是配合積分系統(tǒng)一起使用。本文主要介紹了Redis基于Bitmap實(shí)現(xiàn)用戶簽到功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-06-06