項(xiàng)目中的流程監(jiān)控,有幾種節(jié)點(diǎn),需要監(jiān)控每一個(gè)節(jié)點(diǎn)是否超時(shí)。按傳統(tǒng)的做法,肯定是通過定時(shí)任務(wù),去掃描然后判斷,但是定時(shí)任務(wù)有缺點(diǎn):1,數(shù)據(jù)量大會慢;2,時(shí)間不好控制,太短,怕一次處理不完,太長狀態(tài)就會有延遲。所以就想到用延遲隊(duì)列的方式去實(shí)現(xiàn)。
在redis的配置里把這個(gè)注釋去掉
notify-keyspace-events Ex
然后重啟redis
繼承KeyExpirationEventMessageListener類,實(shí)現(xiàn)父類的方法,就可以監(jiān)聽key過期時(shí)間了。當(dāng)有key過期,就會執(zhí)行這里。這里就把需要的key過濾出來,然后發(fā)送給kafka隊(duì)列。
@Component @Slf4j public class RedisKeyExpirationListener extends KeyExpirationEventMessageListener { @Autowired private KafkaProducerService kafkaProducerService; public RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) { super(listenerContainer); } /** * 針對 redis 數(shù)據(jù)失效事件,進(jìn)行數(shù)據(jù)處理 * @param message * @param pattern */ @Override public void onMessage(Message message, byte[] pattern){ if(message == null || StringUtils.isEmpty(message.toString())){ return; } String content = message.toString(); //key的格式為 flag:時(shí)效類型:運(yùn)單號 示例如下 try { if(content.startsWith(AbnConstant.EMS)){ kafkaProducerService.sendMessageSync(TopicConstant.EMS_WAYBILL_ABN_QUEUE,content); }else if(content.startsWith(AbnConstant.YUNDA)){ kafkaProducerService.sendMessageSync(TopicConstant.YUNDA_WAYBILL_ABN_QUEUE,content); } } catch (Exception e) { log.error("監(jiān)控過期key,發(fā)送kafka異常,",e); } } }
可以看的出來,這種方式其實(shí)是很簡單的,但是有幾個(gè)問題需要注意,一是,這個(gè)盡量單機(jī)運(yùn)行,因?yàn)槎嗯_機(jī)器都會執(zhí)行,浪費(fèi)cpu,增加數(shù)據(jù)庫負(fù)擔(dān)。二是,機(jī)器頻繁部署的時(shí)候,如果有時(shí)間間隔,會出現(xiàn)數(shù)據(jù)的漏處理。
可以看到生產(chǎn)者很簡單,其實(shí)就是利用zset的特性,給一個(gè)zset添加元素而已,而時(shí)間就是它的score。
public void produce(Integer taskId, long exeTime) { System.out.println("加入任務(wù), taskId: " + taskId + ", exeTime: " + exeTime + ", 當(dāng)前時(shí)間:" + LocalDateTime.now()); RedisOps.getJedis().zadd(RedisOps.key, exeTime, String.valueOf(taskId)); }
消費(fèi)者的代碼也不難,就是把已經(jīng)過期的zset中的元素給刪除掉,然后處理數(shù)據(jù)。
public void consumer() { Executors.newSingleThreadExecutor().submit(new Runnable() { @Override public void run() { while (true) { SetString> taskIdSet = RedisOps.getJedis().zrangeByScore(RedisOps.key, 0, System.currentTimeMillis(), 0, 1); if (taskIdSet == null || taskIdSet.isEmpty()) { System.out.println("沒有任務(wù)"); } else { taskIdSet.forEach(id -> { long result = RedisOps.getJedis().zrem(RedisOps.key, id); if (result == 1L) { System.out.println("從延時(shí)隊(duì)列中獲取到任務(wù),taskId:" + id + " , 當(dāng)前時(shí)間:" + LocalDateTime.now()); } }); } try { TimeUnit.MILLISECONDS.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }); }
可以看到這種方式其實(shí)是比上個(gè)方式要好的。因?yàn)?,他的那兩個(gè)缺點(diǎn)都被克服掉了。多臺機(jī)器也沒事兒,也不用再擔(dān)心部署時(shí)間間隔長的問題。
兩個(gè)方式都是不錯(cuò)的,都能解決問題。碰到問題,多思考,多總結(jié)。
到此這篇關(guān)于redis實(shí)現(xiàn)延時(shí)隊(duì)列的兩種方式(小結(jié))的文章就介紹到這了,更多相關(guān)redis 延時(shí)隊(duì)列內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
標(biāo)簽:果洛 朝陽 吉安 江蘇 臺州 大慶 北京 楊凌
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《redis實(shí)現(xiàn)延時(shí)隊(duì)列的兩種方式(小結(jié))》,本文關(guān)鍵詞 redis,實(shí)現(xiàn),延時(shí),隊(duì)列,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。上一篇:redis如何后臺啟動的方法