欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2019年08月27日 09:44:36   作者:雨茗良記  
這篇文章主要給大家介紹了基于redis實現(xiàn)定時任務(wù)的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或者使用redis具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧

前言

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

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

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

技術(shù)棧

redis / nodeJs / koa

技術(shù)重難點

  • 開啟redis的鍵空間通知功能(2.8.0及以上的版本才有此功能)
  • 盡量使用單獨的redis db來實現(xiàn)
  • 使用基于redis的分布式鎖來實現(xiàn)相關(guān)事件不會被重復(fù)消費
  • 需要二次使用的信息需要體現(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') // 存儲實例和訂閱實例需要為兩個不同的實例
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') // 存儲實例設(shè)置為推送鍵過期事件
 console.log('redis init success')
})

subClient.on('ready', () => { // 服務(wù)重啟后依舊可以初始化所有processor
 subClient.subscribe(`__keyevent@${config.redis.sub.db}__:expired`) // 訂閱實例負責(zé)訂閱消息
 subClient.on('message', async (cahnnel, expiredKey) => {
  // 分布式鎖的key不做監(jiān)聽處理
  if (expiredKey.match(innerDistributedLockKeyReg)) return
  // 簡易分布式鎖,拿到鎖的實例消費event
  const cackeKey = `${innerDistributedLockKey}-${expiredKey}`
  const lock = await saveClient.set(cackeKey, 2, 'ex', 5, 'nx') // 這里的用法可以實現(xiàn)簡易的分布式鎖
  if (lock === 'OK') {
   await saveClient.del(cackeKey)
   for (let key in processor) {
    processor[key](expiredKey) // processor對應(yīng)的是接收到相關(guān)鍵過期通知后執(zhí)行的業(yè)務(wù)邏輯,比如推送短信,然后在相關(guān)processor中再次set一個定時過期的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è)置為一段時間后過期,過期后會再次觸發(fā)本邏輯
  }
 }
}

總結(jié)

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

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

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

相關(guān)文章

  • Redis常見限流算法原理及實現(xiàn)

    Redis常見限流算法原理及實現(xiàn)

    這篇文章主要介紹了Redis常見限流算法原理及實現(xiàn),限流簡稱流量限速(Rate?Limit)是指只允許指定的事件進入系統(tǒng),超過的部分將被拒絕服務(wù)、排隊或等待、降級等處理
    2022-08-08
  • redis在php中常用的語法【推薦】

    redis在php中常用的語法【推薦】

    string是redis最基本的類型,而且string類型是二進制安全的。這篇文章主要介紹了redis在php中常用的語法,需要的朋友可以參考下
    2018-08-08
  • 淺析Redis?切片集群的數(shù)據(jù)傾斜問題

    淺析Redis?切片集群的數(shù)據(jù)傾斜問題

    如果?Redis?中的部署,采用的是切片集群,數(shù)據(jù)是會按照一定的規(guī)則分散到不同的實例中保存,比如,使用?Redis?Cluster?或?Codis,這篇文章主要介紹了Redis?切片集群的數(shù)據(jù)傾斜分析,需要的朋友可以參考下
    2022-06-06
  • 淺談Redis處理接口冪等性的兩種方案

    淺談Redis處理接口冪等性的兩種方案

    本文主要介紹了淺談Redis處理接口冪等性的兩種方案,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-08-08
  • Redis分布式限流的幾種實現(xiàn)

    Redis分布式限流的幾種實現(xiàn)

    分布式限流是指通過將限流策略嵌入到分布式系統(tǒng)中,以控制流量或保護服務(wù),本文就來介紹一下Redis分布式限流的幾種實現(xiàn),感興趣的可以了解一下
    2023-12-12
  • Linux下redis5.0.5的安裝過程與配置方法

    Linux下redis5.0.5的安裝過程與配置方法

    這篇文章主要介紹了Linux下redis5.0.5的安裝過程與配置方法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-11-11
  • Redis實現(xiàn)分布式隊列淺析

    Redis實現(xiàn)分布式隊列淺析

    Redis將數(shù)據(jù)存儲在內(nèi)存中,使得讀寫速度非常快,經(jīng)常被用來做緩存系統(tǒng),這里我們將redis用來做一個分布式的消息隊列。這篇文章主要介紹了使用redis來作為消息隊列,并且進行分布式主從配置,有需要的朋友可以參考借鑒,下面來一起看看吧。
    2016-11-11
  • Spring Boot整合Redis實現(xiàn)訂單超時處理問題

    Spring Boot整合Redis實現(xiàn)訂單超時處理問題

    這篇文章主要介紹了Spring Boot整合Redis實現(xiàn)訂單超時處理,通過這個基本的示例,你可以了解如何使用Spring Boot和Redis來處理訂單超時問題,并根據(jù)需要進行擴展和定制,需要的朋友可以參考下
    2023-11-11
  • Redis權(quán)限和訪問控制的實現(xiàn)示例

    Redis權(quán)限和訪問控制的實現(xiàn)示例

    Redis提供了一些機制來保護敏感數(shù)據(jù)和限制對Redis服務(wù)器的訪問,本文主要介紹了Redis權(quán)限和訪問控制的實現(xiàn)示例,具有一定的參考價值,感興趣的可以了解一下
    2023-12-12
  • 關(guān)于redis可視化工具讀取數(shù)據(jù)亂碼問題

    關(guān)于redis可視化工具讀取數(shù)據(jù)亂碼問題

    大家來聊一聊在日常操作redis時用的是什么工具,redis提供的一些命令你都了解了嗎,今天通過本文給大家介紹redis可視化工具讀取數(shù)據(jù)亂碼問題,感興趣的朋友跟隨小編一起看看吧
    2021-07-07

最新評論