欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Laravel隊列實現(xiàn)原理解決問題記錄

 更新時間:2017年08月19日 20:10:04   作者:Dr點燃  
本篇文章主要介紹了淺談Laravel隊列實現(xiàn)原理解決問題記錄,小編覺得挺不錯的,現(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í)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

最新評論