Laravel中為什么不使用blpop取隊(duì)列詳析
前言
Redis 的 list 數(shù)據(jù)結(jié)構(gòu)常用來做消息隊(duì)列,通常使用的命令有 lpop/rpop ,還有帶阻塞版的 blpop/brpop 等。Laravel 5.3 消息隊(duì)列也是用的 lpop 取消息,為什么不用阻塞版的 blpop 呢?
blpop 不用一直輪詢,還可以同時(shí)取多個(gè)隊(duì)列,blpop high low 30,更方便實(shí)現(xiàn)隊(duì)列的優(yōu)先級。
安全隊(duì)列和不安全隊(duì)列
什么是不安全的隊(duì)列?比如客戶端 lpop(統(tǒng)一以 lpop 為例) 從 redis 取出來的 job(任務(wù))還沒處理完進(jìn)程掛掉了或者遇到了異常,由于此時(shí)服務(wù)器上已經(jīng)沒有副本了,這個(gè) job 就丟失了。這種隊(duì)列就是不安全的。
Laravel 正是為了保證消息隊(duì)列的可靠,進(jìn)程掛掉了或者處理失敗還可以重試等,做了比較完善的機(jī)制,如取隊(duì)列的同時(shí)把隊(duì)列放入另一個(gè)集合中“暫存”起來。如代碼所示,使用 lpop 取出隊(duì)列,同時(shí) zadd 到另一個(gè)集合,使用 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 隊(duì)列工作原理之前有一篇博文進(jìn)行了整理,請參考:http://chabaoo.cn/article/131414.htm
為什么不用 blpop?
這里為什么不使用阻塞版本的 blpop 呢?
blpop 是阻塞版的 lpop,如果隊(duì)列沒有數(shù)據(jù)過來,那么在超時(shí)時(shí)間內(nèi)就會一直阻塞,直到 rpush 數(shù)據(jù)到隊(duì)列,有點(diǎn)類似 http 的長輪詢,假如客戶端取出數(shù)據(jù)的這一刻掛了,還沒來得及暫存到另外的集合中,那么這個(gè)數(shù)據(jù)就丟失了。
你可能會問為何不跟 lpop 一樣用 lua 腳本來處理并保證原子性?這個(gè)問題作者在 github 上有回答。(https://github.com/laravel/framework/issues/22939)

我們知道 redis lua 腳本實(shí)際上就是事務(wù),作者的大意也是說 MULTI/EXEC 包裹起來的 blpop 沒有意義,這個(gè)時(shí)候它“退化”為非阻塞版的。
Redis 官方文檔也有說明:
在MULTI/EXEC事務(wù)中的BLPOP
BLPOP 可以用于流水線(pipline,批量地發(fā)送多個(gè)命令并讀入多個(gè)回復(fù)),但把它用在 MULTI / EXEC 塊當(dāng)中沒有意義。因?yàn)檫@要求整個(gè)服務(wù)器被阻塞以保證塊執(zhí)行時(shí)的原子性,該行為阻止了其他客戶端執(zhí)行 LPUSH 或 RPUSH 命令。
因此,一個(gè)被包裹在 MULTI / EXEC 塊內(nèi)的 BLPOP 命令,行為表現(xiàn)得就像 LPOP 一樣,對空列表返回 nil ,對非空列表彈出列表元素,不進(jìn)行任何阻塞操作。
因此通過 lua 腳本操作 blpop 和 zadd 也沒有意義,結(jié)論就是:因?yàn)闆]用到阻塞的特性,或者無法保證原子性。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
- laravel5.6 框架郵件隊(duì)列database驅(qū)動簡單demo示例
- Laravel使用Queue隊(duì)列的技巧匯總
- Docker部署Laravel應(yīng)用實(shí)現(xiàn)隊(duì)列&任務(wù)調(diào)度
- Laravel 6 將新增為指定隊(duì)列任務(wù)設(shè)置中間件的功能
- Laravel 隊(duì)列使用的實(shí)現(xiàn)
- Laravel框架隊(duì)列原理與用法分析
- 關(guān)于 Laravel Redis 多個(gè)進(jìn)程同時(shí)取隊(duì)列問題詳解
- 源碼分析 Laravel 重復(fù)執(zhí)行同一個(gè)隊(duì)列任務(wù)的原因
- Laravel使用消息隊(duì)列需要注意的一些問題
- Laravel中利用隊(duì)列發(fā)送郵件的方法示例
- 淺談Laravel隊(duì)列實(shí)現(xiàn)原理解決問題記錄
- 淺析Laravel5中隊(duì)列的配置及使用
- PHP的Laravel框架中使用消息隊(duì)列queue及異步隊(duì)列的方法
- Laravel 4.2 中隊(duì)列服務(wù)(queue)使用感受
- Laravel框架中隊(duì)列和工作(Queues、Jobs)操作實(shí)例詳解
相關(guān)文章
PHP性能優(yōu)化工具篇Benchmark類調(diào)試執(zhí)行時(shí)間
這是PHP性能優(yōu)化系列第二期,如何使用PEAR工具類Benchmark逐行獲取代碼或函數(shù)的執(zhí)行時(shí)間2011-12-12
PHP5中實(shí)現(xiàn)多態(tài)的兩種方法實(shí)例分享
多態(tài)這個(gè)概念,在Java中指的是變量可以指向的對象的類型,可是變量聲明類型的子類。對象一旦創(chuàng)建,它的類型是不變的,多態(tài)的是變量2014-04-04
php框架CodeIgniter主從數(shù)據(jù)庫配置方法分析
這篇文章主要介紹了php框架CodeIgniter主從數(shù)據(jù)庫配置方法,結(jié)合實(shí)例形式分析了CodeIgniter框架主從數(shù)據(jù)庫配置方法、模型model與控制器調(diào)用操作技巧以及相關(guān)注意事項(xiàng),需要的朋友可以參考下2018-05-05
PHP實(shí)現(xiàn)的交通銀行網(wǎng)銀在線支付接口ECSHOP插件和使用例子
這篇文章主要介紹了PHP實(shí)現(xiàn)的交通銀行網(wǎng)銀在線支付接口ECSHOP插件和使用例子,需要的朋友可以參考下2014-05-05
淺談PHP攔截器之__set()與__get()的理解與使用方法
下面小編就為大家?guī)硪黄獪\談PHP攔截器之__set()與__get()的理解與使用方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10
smarty模板引擎之配置文件數(shù)據(jù)和保留數(shù)據(jù)
這篇文章主要介紹了smarty模板引擎之配置文件數(shù)據(jù)和保留數(shù)據(jù)的方法,實(shí)例分析了smarty模板引擎配置文件數(shù)據(jù)及獲取數(shù)據(jù)的具體技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
PHP創(chuàng)建PowerPoint2007文檔的方法
這篇文章主要介紹了PHP創(chuàng)建PowerPoint2007文檔的方法,通過PHP第三方插件PHPPowerPoint類庫實(shí)現(xiàn)ppt文件的生成功能,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-12-12

