婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識(shí)庫 > 基于redis實(shí)現(xiàn)定時(shí)任務(wù)的方法詳解

基于redis實(shí)現(xiàn)定時(shí)任務(wù)的方法詳解

熱門標(biāo)簽:四川穩(wěn)定外呼系統(tǒng)軟件 b2b外呼系統(tǒng) 一個(gè)地圖標(biāo)注多少錢 400電話辦理的口碑 南京手機(jī)外呼系統(tǒng)廠家 廊坊外呼系統(tǒng)在哪買 地圖標(biāo)注工廠入駐 高碑店市地圖標(biāo)注app 臺(tái)灣電銷

前言

業(yè)務(wù)中碰到的需求(抽象描述一下):針對(duì)不同的用戶能夠?qū)崿F(xiàn)不同時(shí)間的間隔循環(huán)任務(wù)。比如在用戶注冊成功24小時(shí)后給用戶推送相關(guān)短信等類似需求。

使用crontab?太重,且基本不現(xiàn)實(shí),不可能給每一個(gè)用戶在服務(wù)器上生成一個(gè)定時(shí)任務(wù)。
定時(shí)輪詢?IO頻繁且效率太低

想到經(jīng)常的使用的redis可以設(shè)置緩存時(shí)間,應(yīng)該會(huì)有過期的事件通知吧,查了一下文檔,果然有相關(guān)配置,叫做“鍵空間事件通知”。具體說明可參考官方文檔。

技術(shù)棧

redis / nodeJs / koa

技術(shù)重難點(diǎn)

  • 開啟redis的鍵空間通知功能(2.8.0及以上的版本才有此功能)
  • 盡量使用單獨(dú)的redis db來實(shí)現(xiàn)
  • 使用基于redis的分布式鎖來實(shí)現(xiàn)相關(guān)事件不會(huì)被重復(fù)消費(fèi)
  • 需要二次使用的信息需要體現(xiàn)在redis緩存的key中
  • redis cache key使用業(yè)務(wù)前綴,避免重名覆蓋
  • 防止業(yè)務(wù)服務(wù)重啟導(dǎo)致nodejs層面的監(jiān)聽失效

"talk is cheap, show me the code 🤖"

核心代碼

核心代碼
const { saveClient, subClient } = require('./db/redis') // 存儲(chǔ)實(shí)例和訂閱實(shí)例需要為兩個(gè)不同的實(shí)例
const processor = require('./service/task')
const config = require('./config/index')
const innerDistributedLockKey = '__' // 內(nèi)部使用的分布式鎖的key的特征值
const innerDistributedLockKeyReg = new RegExp(`^${innerDistributedLockKey}`)

saveClient.on('ready', async () => {
 saveClient.config('SET', 'notify-keyspace-events', 'Ex') // 存儲(chǔ)實(shí)例設(shè)置為推送鍵過期事件
 console.log('redis init success')
})

subClient.on('ready', () => { // 服務(wù)重啟后依舊可以初始化所有processor
 subClient.subscribe(`__keyevent@${config.redis.sub.db}__:expired`) // 訂閱實(shí)例負(fù)責(zé)訂閱消息
 subClient.on('message', async (cahnnel, expiredKey) => {
  // 分布式鎖的key不做監(jiān)聽處理
  if (expiredKey.match(innerDistributedLockKeyReg)) return
  // 簡易分布式鎖,拿到鎖的實(shí)例消費(fèi)event
  const cackeKey = `${innerDistributedLockKey}-${expiredKey}`
  const lock = await saveClient.set(cackeKey, 2, 'ex', 5, 'nx') // 這里的用法可以實(shí)現(xiàn)簡易的分布式鎖
  if (lock === 'OK') {
   await saveClient.del(cackeKey)
   for (let key in processor) {
    processor[key](expiredKey) // processor對(duì)應(yīng)的是接收到相關(guān)鍵過期通知后執(zhí)行的業(yè)務(wù)邏輯,比如推送短信,然后在相關(guān)processor中再次set一個(gè)定時(shí)過期的key
   }
  }
 })
 console.log('subClient init success')
})
servide/task (processor)
exports.sendMessage = async function sendMessage(expiredKey, subClient) {
 // 只處理相關(guān)業(yè)務(wù)的過期事件
 if (expiredKey.match(/^send_message/)) {
  const [prefix, userId, type] = expiredKey.split('-')
  let user = getUser(userId)
  if (user.phone) {
   push(message) // 偽代碼
   resetRedisKey(expiredKey, ttl) // 重新把key設(shè)置為一段時(shí)間后過期,過期后會(huì)再次觸發(fā)本邏輯
  }
 }
}

總結(jié)

  • 此功能利用了redis的鍵空間通知功能實(shí)現(xiàn)了簡單了基于用戶或者基于不同業(yè)務(wù)場景的定時(shí)任務(wù)功能。由于鍵空間事件通知功能是一個(gè)較消耗CPU的操作,所以建議使用單獨(dú)的DB來處理。
  • 這里展示出來的是基本用法,未考慮定時(shí)任務(wù)的持久化功能,如果使用過程中redis故障重啟,則會(huì)導(dǎo)致所有定時(shí)任務(wù)丟失。如果在redis發(fā)布鍵失效通知時(shí),訂閱服務(wù)出故障未在線,或者網(wǎng)絡(luò)問題沒有被消費(fèi)方收到,也會(huì)導(dǎo)致此次事件丟失。
  • redis的expired事件并不是在key過期的時(shí)候觸發(fā),而是在key被刪除的時(shí)候觸發(fā)。redis會(huì)定期清理過期的key,或者當(dāng)訪問key的時(shí)候檢查是否過期,只有這時(shí)過期的key才會(huì)觸發(fā)刪除操作,因此會(huì)有一些小的時(shí)間差距(個(gè)人的實(shí)際使用中并沒有影響用戶體驗(yàn))。

因此需要權(quán)衡使用redis的過期機(jī)制實(shí)現(xiàn)的定時(shí)任務(wù)的使用場景。

好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。

您可能感興趣的文章:
  • java操作Redis緩存設(shè)置過期時(shí)間的方法
  • 使用redis的increment()方法實(shí)現(xiàn)計(jì)數(shù)器功能案例
  • Redis有效時(shí)間設(shè)置以及時(shí)間過期處理操作

標(biāo)簽:甘南 河源 拉薩 南寧 畢節(jié) 定州 伊春 泰州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《基于redis實(shí)現(xiàn)定時(shí)任務(wù)的方法詳解》,本文關(guān)鍵詞  基于,redis,實(shí)現(xiàn),定時(shí),任務(wù),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《基于redis實(shí)現(xiàn)定時(shí)任務(wù)的方法詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于基于redis實(shí)現(xiàn)定時(shí)任務(wù)的方法詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    主站蜘蛛池模板: 内丘县| 庐江县| 奉化市| 阳春市| 浦东新区| 临漳县| 嵊泗县| 齐齐哈尔市| 凉城县| 垫江县| 新兴县| 永新县| 桂东县| 务川| 中山市| 兴海县| 敖汉旗| 闵行区| 方山县| 伊金霍洛旗| 兴隆县| 抚宁县| 河池市| 固阳县| 独山县| 郧西县| 青岛市| 南开区| 洞口县| 镇赉县| 大荔县| 郎溪县| 上林县| 平罗县| 乌鲁木齐县| 嘉荫县| 额济纳旗| 永州市| 和龙市| 泽库县| 石泉县|