国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 語言 > PHP > 正文

淺談Laravel隊列實現原理解決問題記錄

2024-05-04 23:59:35
字體:
來源:轉載
供稿:網友

問題

公司項目使用Laravel的開發的兩個項目在同一個測試服務器部署,公用同一個redis。在使用laravel中的隊列時,產生沖突干擾。

查找問題原因

在laravel 隊列的操作類Illuminate/Queue/RedisQueue.php中可以看到pushRaw()方法:

// 將一任務推入隊列中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實現是通過list結構實現的,rpush(key, value)是將value推入鍵值為key的redis隊列,key的值則是通過$this->getQueue($queue) 獲取到的

protected function getQueue($queue)  {    return 'queues:'.($queue ?: $this->default);  }

所以的redis中list中的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,所以當共用redis時,默認的隊列list 都是'queue:default',所以導致了沖突。

因為隊列監聽 監聽的隊列名稱是由 --queue參數決定的,如果不傳就是我們上面設置的默認值,若傳了就會根據傳入的隊列名從前往后優先依次處理,具體見代碼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=傳入的參數,當 $queue不存在是直接調用$connection->pop()當參數存在時會將參數解析,優先處理排在前面的隊列名稱,將隊列名稱傳入pop($queue), pop()會嘗試從指定隊列或默認隊列中獲取隊列任務

// Illuminate/Queue/RedisQueue.phppublic 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);    }  }

至此搞清了隊列執行的原理。

解決方法

將queue的配置文件中默認隊列修改為不同的名稱,比如: 'queue' => laravel1','queue' => laravel2'。

隊列監聽 php artisan queue:listen redis --queue=laravel1,syncExpress

最后

遇到問題,莫要病急亂投醫。從代碼入手,分析理解實現原理,找對點,解決方法也許很簡單,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到PHP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 化州市| 原平市| 广丰县| 钟山县| 沽源县| 犍为县| 米泉市| 那曲县| 留坝县| 内黄县| 大关县| 香格里拉县| 罗田县| 延安市| 巴塘县| 胶南市| 和顺县| 科技| 长寿区| 彭泽县| 天峨县| 满洲里市| 双柏县| 博湖县| 宝清县| 庆云县| 南城县| 安徽省| 宜黄县| 巴彦淖尔市| 沂水县| 南丰县| 寿阳县| 桑日县| 绥中县| 黎城县| 烟台市| 徐汇区| 绥棱县| 锦州市| 宜兰市|