淺談Laravel隊(duì)列實(shí)現(xiàn)原理解決問題記錄
問題
公司項(xiàng)目使用Laravel的開發(fā)的兩個(gè)項(xiàng)目在同一個(gè)測試服務(wù)器部署,公用同一個(gè)redis。在使用laravel中的隊(duì)列時(shí),產(chǎn)生沖突干擾。
查找問題原因
在laravel 隊(duì)列的操作類Illuminate\Queue\RedisQueue.php中可以看到pushRaw()方法:
// 將一任務(wù)推入隊(duì)列中
public function pushRaw($payload, $queue = null, array $options = [])
{
$this->getConnection()->rpush($this->getQueue($queue), $payload);
return Arr::get(json_decode($payload, true), 'id');
}
從該方法中可以看出Lrarvel隊(duì)列的redis實(shí)現(xiàn)是通過list結(jié)構(gòu)實(shí)現(xiàn)的,rpush(key, value)是將value推入鍵值為key的redis隊(duì)列,key的值則是通過$this->getQueue($queue) 獲取到的
protected function getQueue($queue)
{
return 'queues:'.($queue ?: $this->default);
}
所以的redis中l(wèi)ist中的key是 'queues:'.($queue ?: $this->default);拼接的,$this->default 的值是 RedisQueue 實(shí)例化的時(shí)候從config\queue.php配置中加載的 'queue' => 'default',$queue 是添加隊(duì)列時(shí)$this->dispatch( new jobClass()->onQueue($queue) )傳入的。
// config\queue.php 文件中的redis配置部分
'redis' => [
'driver' => 'redis',
'connection' => 'default',
'queue' => 'default',
'expire' => 60,
],
至此,兩個(gè)項(xiàng)目的隊(duì)列沖突原因就找到了。因?yàn)閞edis隊(duì)列配置中 'queue' => 'default' 都使用的默認(rèn)的default,所以當(dāng)共用redis時(shí),默認(rèn)的隊(duì)列l(wèi)ist 都是'queue:default',所以導(dǎo)致了沖突。
因?yàn)殛?duì)列監(jiān)聽 監(jiān)聽的隊(duì)列名稱是由 --queue參數(shù)決定的,如果不傳就是我們上面設(shè)置的默認(rèn)值,若傳了就會(huì)根據(jù)傳入的隊(duì)列名從前往后優(yōu)先依次處理,具體見代碼Illuminate\Queue\Worker.php中:
protected function getNextJob($connection, $queue)
{
if (is_null($queue)) {
return $connection->pop();
}
foreach (explode(',', $queue) as $queue) {
if (! is_null($job = $connection->pop($queue))) {
return $job;
}
}
}
$queue就是--queue=傳入的參數(shù),當(dāng) $queue不存在是直接調(diào)用$connection->pop()當(dāng)參數(shù)存在時(shí)會(huì)將參數(shù)解析,優(yōu)先處理排在前面的隊(duì)列名稱,將隊(duì)列名稱傳入pop($queue), pop()會(huì)嘗試從指定隊(duì)列或默認(rèn)隊(duì)列中獲取隊(duì)列任務(wù)
// Illuminate\Queue\RedisQueue.php
public function pop($queue = null)
{
$original = $queue ?: $this->default;
$queue = $this->getQueue($queue);
if (! is_null($this->expire)) {
$this->migrateAllExpiredJobs($queue);
}
$job = $this->getConnection()->lpop($queue);
if (! is_null($job)) {
$this->getConnection()->zadd($queue.':reserved', $this->getTime() + $this->expire, $job);
return new RedisJob($this->container, $this, $job, $original);
}
}
至此搞清了隊(duì)列執(zhí)行的原理。
解決方法
將queue的配置文件中默認(rèn)隊(duì)列修改為不同的名稱,比如: 'queue' => laravel1','queue' => laravel2'。
隊(duì)列監(jiān)聽 php artisan queue:listen redis --queue=laravel1,syncExpress
最后
遇到問題,莫要病急亂投醫(yī)。從代碼入手,分析理解實(shí)現(xiàn)原理,找對點(diǎn),解決方法也許很簡單,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- laravel5.6 框架郵件隊(duì)列database驅(qū)動(dòng)簡單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ì)列原理與用法分析
- Laravel中為什么不使用blpop取隊(duì)列詳析
- 關(guān)于 Laravel Redis 多個(gè)進(jìn)程同時(shí)取隊(duì)列問題詳解
- 源碼分析 Laravel 重復(fù)執(zhí)行同一個(gè)隊(duì)列任務(wù)的原因
- Laravel使用消息隊(duì)列需要注意的一些問題
- Laravel中利用隊(duì)列發(fā)送郵件的方法示例
- 淺析Laravel5中隊(duì)列的配置及使用
- PHP的Laravel框架中使用消息隊(duì)列queue及異步隊(duì)列的方法
- Laravel 4.2 中隊(duì)列服務(wù)(queue)使用感受
- Laravel框架中隊(duì)列和工作(Queues、Jobs)操作實(shí)例詳解
相關(guān)文章
thinkPHP實(shí)現(xiàn)的省市區(qū)三級聯(lián)動(dòng)功能示例
這篇文章主要介紹了thinkPHP實(shí)現(xiàn)的省市區(qū)三級聯(lián)動(dòng)功能,詳細(xì)分析了thinkPHP實(shí)現(xiàn)省市區(qū)三級聯(lián)動(dòng)功能的詳細(xì)步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-05-05
Yii2針對指定url的生成及圖片等的引入方法小結(jié)
這篇文章主要介紹了Yii2針對指定url的生成及圖片等的引入方法,針對常用的URL格式及圖片路徑操作結(jié)合實(shí)例進(jìn)行了對比分析,需要的朋友可以參考下2016-07-07
yii2-GridView在開發(fā)中常用的功能及技巧總結(jié)
本篇文章主要介紹了yii2-GridView在開發(fā)中常用的功能及技巧總結(jié),數(shù)據(jù)網(wǎng)格或者說 GridView 小部件是Yii中最強(qiáng)大的部件之一。有興趣的可以了解一下。2017-01-01
ThinkPHP獨(dú)立分組使用的注意事項(xiàng)
這篇文章主要介紹了ThinkPHP獨(dú)立分組使用的注意事項(xiàng),針對獨(dú)立分組的目錄結(jié)構(gòu)與分組之間的相互調(diào)用進(jìn)行了較為深入的分析,并指出了使用時(shí)的注意事項(xiàng),需要的朋友可以參考下2014-11-11
php arsort 數(shù)組降序排序詳細(xì)介紹
php arsort函數(shù)用于將數(shù)組中的元素按照降序進(jìn)行排序,如果排序成功則返回true,否則返回false,本文章向大家講解arsort函數(shù)的基本語法及使用實(shí)例,需要的朋友可以參考下2016-11-11
Laravel框架集成UEditor編輯器的方法圖文與實(shí)例詳解
這篇文章主要介紹了Laravel框架集成UEditor編輯器的方法,結(jié)合圖文與實(shí)例形式詳細(xì)分析了Laravel框架整合集成UEditor編輯器的相關(guān)操作步驟與具體實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-04-04

