濮阳杆衣贸易有限公司

主頁 > 知識庫 > Laravel中為什么不使用blpop取隊列詳析

Laravel中為什么不使用blpop取隊列詳析

熱門標簽:合肥企業(yè)外呼系統(tǒng)線路 太原400電話申請流程 萍鄉(xiāng)商鋪地圖標注 代理打電話機器人 電信外呼系統(tǒng)多少錢一個月 桂陽公司如何做地圖標注 企業(yè)400電話辦理多少費用 神龍斗士電話機器人 宿州正規(guī)外呼系統(tǒng)軟件

前言

Redis 的 list 數(shù)據(jù)結構常用來做消息隊列,通常使用的命令有 lpop/rpop ,還有帶阻塞版的 blpop/brpop 等。Laravel 5.3 消息隊列也是用的 lpop 取消息,為什么不用阻塞版的 blpop 呢?

blpop 不用一直輪詢,還可以同時取多個隊列,blpop high low 30,更方便實現(xiàn)隊列的優(yōu)先級。

安全隊列和不安全隊列

什么是不安全的隊列?比如客戶端 lpop(統(tǒng)一以 lpop 為例) 從 redis 取出來的 job(任務)還沒處理完進程掛掉了或者遇到了異常,由于此時服務器上已經(jīng)沒有副本了,這個 job 就丟失了。這種隊列就是不安全的。

Laravel 正是為了保證消息隊列的可靠,進程掛掉了或者處理失敗還可以重試等,做了比較完善的機制,如取隊列的同時把隊列放入另一個集合中“暫存”起來。如代碼所示,使用 lpop 取出隊列,同時 zadd 到另一個集合,使用 redis lua 來保證原子性。

public static function pop()
{
 return 'LUA'
-- Pop the first job off of the queue...
local job = redis.call('lpop', KEYS[1])
local reserved = false
 
if(job ~= false) then
-- Increment the attempt count and place job on the reserved queue...
reserved = cjson.decode(job)
reserved['attempts'] = reserved['attempts'] + 1
reserved = cjson.encode(reserved)
redis.call('zadd', KEYS[2], ARGV[1], reserved)
end
 
return {job, reserved}
LUA;
}

具體 Laravel 隊列工作原理之前有一篇博文進行了整理,請參考:https://www.jb51.net/article/131414.htm

為什么不用 blpop?

這里為什么不使用阻塞版本的 blpop 呢?

blpop 是阻塞版的 lpop,如果隊列沒有數(shù)據(jù)過來,那么在超時時間內(nèi)就會一直阻塞,直到 rpush 數(shù)據(jù)到隊列,有點類似 http 的長輪詢,假如客戶端取出數(shù)據(jù)的這一刻掛了,還沒來得及暫存到另外的集合中,那么這個數(shù)據(jù)就丟失了。

你可能會問為何不跟 lpop 一樣用 lua 腳本來處理并保證原子性?這個問題作者在 github 上有回答。(https://github.com/laravel/framework/issues/22939)

我們知道 redis lua 腳本實際上就是事務,作者的大意也是說 MULTI/EXEC 包裹起來的 blpop 沒有意義,這個時候它“退化”為非阻塞版的。

Redis 官方文檔也有說明:

在MULTI/EXEC事務中的BLPOP

BLPOP 可以用于流水線(pipline,批量地發(fā)送多個命令并讀入多個回復),但把它用在 MULTI / EXEC 塊當中沒有意義。因為這要求整個服務器被阻塞以保證塊執(zhí)行時的原子性,該行為阻止了其他客戶端執(zhí)行 LPUSH 或 RPUSH 命令。

因此,一個被包裹在 MULTI / EXEC 塊內(nèi)的 BLPOP 命令,行為表現(xiàn)得就像 LPOP 一樣,對空列表返回 nil ,對非空列表彈出列表元素,不進行任何阻塞操作。

因此通過 lua 腳本操作 blpop 和 zadd 也沒有意義,結論就是:因為沒用到阻塞的特性,或者無法保證原子性。

總結

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • laravel5.6 框架郵件隊列database驅動簡單demo示例
  • Laravel使用Queue隊列的技巧匯總
  • Docker部署Laravel應用實現(xiàn)隊列&任務調(diào)度
  • Laravel 6 將新增為指定隊列任務設置中間件的功能
  • Laravel 隊列使用的實現(xiàn)
  • Laravel框架隊列原理與用法分析
  • 關于 Laravel Redis 多個進程同時取隊列問題詳解
  • 源碼分析 Laravel 重復執(zhí)行同一個隊列任務的原因
  • Laravel使用消息隊列需要注意的一些問題
  • Laravel中利用隊列發(fā)送郵件的方法示例
  • 淺談Laravel隊列實現(xiàn)原理解決問題記錄
  • 淺析Laravel5中隊列的配置及使用
  • PHP的Laravel框架中使用消息隊列queue及異步隊列的方法
  • Laravel 4.2 中隊列服務(queue)使用感受
  • Laravel框架中隊列和工作(Queues、Jobs)操作實例詳解

標簽:白銀 廊坊 衡陽 綏化 太原 崇左 辛集 鄂州

巨人網(wǎng)絡通訊聲明:本文標題《Laravel中為什么不使用blpop取隊列詳析》,本文關鍵詞  Laravel,中,為什么,不,使用,;如發(fā)現(xiàn)本文內(nèi)容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Laravel中為什么不使用blpop取隊列詳析》相關的同類信息!
  • 本頁收集關于Laravel中為什么不使用blpop取隊列詳析的相關信息資訊供網(wǎng)民參考!
  • 推薦文章
    陆丰市| 县级市| 台湾省| 辽中县| 漳浦县| 鞍山市| 田东县| 额尔古纳市| 白水县| 濮阳市| 永顺县| 澄城县| 宁海县| 石首市| 平塘县| 兴海县| 南和县| 琼结县| 扎鲁特旗| 呼和浩特市| 古田县| 高雄县| 外汇| 仙桃市| 邯郸县| 临沂市| 江源县| 通渭县| 射洪县| 荥阳市| 偃师市| 日喀则市| 调兵山市| 永新县| 吉隆县| 罗田县| 汤阴县| 阳山县| 建瓯市| 鄢陵县| 闻喜县|