成人精品一区二区三区中文字幕-成人精品一区二区三区-成人精品一级毛片-成人精品亚洲-日本在线视频一区二区-日本在线视频免费

導航首頁 ? 技術教程 ? 淺談Laravel隊列實現原理解決問題記錄
全站頭部文字 我要出現在這里
淺談Laravel隊列實現原理解決問題記錄 689 2023-12-10   

問題

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

查找問題原因

在laravel 隊列的操作類IlluminateQueueRedisQueue.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 實例化的時候從configqueue.php配置中加載的 'queue' => 'default',$queue 是添加隊列時$this->dispatch( new jobClass()->onQueue($queue) )傳入的。

// configqueue.php 文件中的redis配置部分
'redis' => [
      'driver'   => 'redis',
      'connection' => 'default',
      'queue'   => 'default',
      'expire'   => 60,
    ],

至此,兩個項目的隊列沖突原因就找到了。因為redis隊列配置中 'queue' => 'default' 都使用的默認的default,所以當共用redis時,默認的隊列list 都是'queue:default',所以導致了沖突。

因為隊列監聽 監聽的隊列名稱是由 --queue參數決定的,如果不傳就是我們上面設置的默認值,若傳了就會根據傳入的隊列名從前往后優先依次處理,具體見代碼IlluminateQueueWorker.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()會嘗試從指定隊列或默認隊列中獲取隊列任務

// IlluminateQueueRedisQueue.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);
    }
  }

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

解決方法

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

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

最后

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



主站蜘蛛池模板: 小松未可子| 冒险王2| 法律援助中心免费写诉状| 幼儿园一日活动的组织与实施| 新闻联播台词| 女生操| 健步如飞的蜗牛三年级作文| 《stag》电影在线观看| 根深蒂固韩国电影| 韩国电影《我是谁》演员表介绍| 痛风能吃的菜一览表| 王若晰 个人资料| 黄老汉| 战长沙每个人的结局| 共同财产电影| 吴燕妮个人资料简介| 哈利学前班| 大石桥联盟| 十三刺客| 尘埃落定剧情| 水中生孩子视频| 电影《天启》| 电车摩女| 成毅壁纸| 陈宛蔚| deaf dj课文翻译| 秃探与俏妞| 抖音手机版| 谭老板 电影| 雾里简谱| 黄网站在线观看视频| 最贵的香烟| 大珍珠演员表介绍| 双世宠妃1| 安德鲁·林肯| 中秋节的作文| 敦煌夜谭在线观看| bangdream动漫| 晋剧下河东全本| 欧美乱淫av片免费黑鬼| 少年派2 2022 张嘉益|

!!!站長長期在線接?。?!

網站、小程序:定制開發/二次開發/仿制開發等

各種疑難雜癥解決/定制接口/定制采集等

站長微信:lxwl520520

站長QQ:1737366103