淺談Laravel隊列實現(xiàn)原理解決問題記錄
問題
公司項目使用Laravel的開發(fā)的兩個項目在同一個測試服務(wù)器部署,公用同一個redis。在使用laravel中的隊列時,產(chǎn)生沖突干擾。
查找問題原因
在laravel 隊列的操作類Illuminate\Queue\RedisQueue.php
中可以看到pushRaw()
方法:
// 將一任務(wù)推入隊列中 public function pushRaw($payload, $queue = null, array $options = []) { $this->getConnection()->rpush($this->getQueue($queue), $payload); return Arr::get(json_decode($payload, true), 'id'); }
從該方法中可以看出Lrarvel隊列的redis實現(xiàn)是通過list結(jié)構(gòu)實現(xiàn)的,rpush(key, value)
是將value推入鍵值為key的redis隊列,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
實例化的時候從config\queue.php
配置中加載的 'queue' => 'default'
,$queue 是添加隊列時$this->dispatch( new jobClass()->onQueue($queue) )
傳入的。
// config\queue.php 文件中的redis配置部分 'redis' => [ 'driver' => 'redis', 'connection' => 'default', 'queue' => 'default', 'expire' => 60, ],
至此,兩個項目的隊列沖突原因就找到了。因為redis隊列配置中 'queue' => 'default'
都使用的默認的default,所以當(dāng)共用redis時,默認的隊列l(wèi)ist 都是'queue:default',所以導(dǎo)致了沖突。
因為隊列監(jiān)聽 監(jiān)聽的隊列名稱是由 --queue參數(shù)決定的,如果不傳就是我們上面設(shè)置的默認值,若傳了就會根據(jù)傳入的隊列名從前往后優(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ù)解析,優(yōu)先處理排在前面的隊列名稱,將隊列名稱傳入pop($queue), pop()
會嘗試從指定隊列或默認隊列中獲取隊列任務(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); } }
至此搞清了隊列執(zhí)行的原理。
解決方法
將queue的配置文件中默認隊列修改為不同的名稱,比如: 'queue' => laravel1','queue' => laravel2'。
隊列監(jiān)聽 php artisan queue:listen redis --queue=laravel1,syncExpress
最后
遇到問題,莫要病急亂投醫(yī)。從代碼入手,分析理解實現(xiàn)原理,找對點,解決方法也許很簡單,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- laravel5.6 框架郵件隊列database驅(qū)動簡單demo示例
- Laravel使用Queue隊列的技巧匯總
- Docker部署Laravel應(yīng)用實現(xiàn)隊列&任務(wù)調(diào)度
- Laravel 6 將新增為指定隊列任務(wù)設(shè)置中間件的功能
- Laravel 隊列使用的實現(xiàn)
- Laravel框架隊列原理與用法分析
- Laravel中為什么不使用blpop取隊列詳析
- 關(guān)于 Laravel Redis 多個進程同時取隊列問題詳解
- 源碼分析 Laravel 重復(fù)執(zhí)行同一個隊列任務(wù)的原因
- Laravel使用消息隊列需要注意的一些問題
- Laravel中利用隊列發(fā)送郵件的方法示例
- 淺析Laravel5中隊列的配置及使用
- PHP的Laravel框架中使用消息隊列queue及異步隊列的方法
- Laravel 4.2 中隊列服務(wù)(queue)使用感受
- Laravel框架中隊列和工作(Queues、Jobs)操作實例詳解
相關(guān)文章
PHP實現(xiàn)的簡單留言板功能示例【基于thinkPHP框架】
這篇文章主要介紹了PHP實現(xiàn)的簡單留言板功能,結(jié)合實例形式分析了基于thinkPHP框架實現(xiàn)的留言板相關(guān)配置、數(shù)據(jù)庫操作、sql語句等實現(xiàn)技巧與操作注意事項,需要的朋友可以參考下2018-12-12TP框架實現(xiàn)上傳一張圖片和批量上傳圖片的方法分析
這篇文章主要介紹了TP框架實現(xiàn)上傳一張圖片和批量上傳圖片的方法,結(jié)合實例形式分析了TP框架圖片上傳操作相關(guān)原理、實現(xiàn)步驟及操作注意事項,需要的朋友可以參考下2020-04-04php側(cè)拉菜單 漂亮,可以向右或者向左展開,支持FF,IE
其中menu UL UL的left或者right控制向右或者向左顯示子菜單2009-10-10