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

首頁 > 開發 > PHP > 正文

Laravel 4.2 中隊列服務(queue)使用感受

2024-05-04 23:26:32
字體:
來源:轉載
供稿:網友
Laravel的隊列組件為許多隊列服務提供了統一的API接口。隊列服務讓你可以異步處理一個耗時任務,比如延遲發送一封郵件,從而大大加快了應用的Web請求處理速度。
 
 

這半個月,我參與重寫了一個微信公眾號后端系統,首次使用了laravel 4.2,以及laravel引以為傲的隊列服務(queue)。

由于整個系統涉及到多端交互,又有大量語音傳輸、處理的業務,我們在一些地方發現響應時間過長。之前的系統基于node.js和mongoDB,由于node天生就是異步,有守護進程,所以并沒有出現過這個問題,而這次重寫必然要引入異步流程了。Queue進入了我們的視線。

根據這一頁幾乎還全是英文的”中文文檔“ ,laravel恰好在4.2版本中剛剛引入了redis作為隊列存儲,這是一個非常好的消息。OK,背景介紹到這里,下面扯扯干貨。

laravel中的隊列服務跟其他隊列服務也沒有什么不同,都是最符合人類思維的最簡單最普遍的流程:有一個地方存放隊列信息,一個PHP進程在運行時將任務寫入,另外一個PHP守護進程輪詢隊列信息,將達到執行要求的任務執行并刪除。由于PHP是url驅動的同步語言,本身是阻塞的,所以laravel提供一個守護進程工具來查詢并執行隊列信息也就不足為奇了。

Laravel的queue配置文件是 /app/config/queue.php,在 Default Queue Driver 這一項中,可以選擇"sync", "beanstalkd", "sqs", "iron", "redis" 五種驅動器。

1. sync是本地調試用的同步驅動器

2. beanstalkd 是一個專業隊列服務驅動器:http://kr.github.io/beanstalkd/

3. sqs和iron是國外第三方隊列服務

4. 最后一項redis給了我們一個使用redis的理由,這樣我們順便把緩存服務和session服務全部遷移到redis上了。

0. 順便說一句,session驅動器千萬別用mysql,處理時間1S不是夢,哎,看誰呢,說的就是你,1S哥!

隊列服務需要專門新建任務類,作為獨立類,他們不需要繼承類,因為隊列里的任務在執行的時候,是由PHP守護進程來獨立調用的,當然如果你要use一下別的類再調用,也不會出錯。之前我把很多額外服務獨立到了一個單獨的文件夾 /app/services 里,比如輸入信息驗證 validator,特殊安全驗證模塊等,這次queue類們就位于其中。

queue的使用非常簡單,下面就是一個簡單的示例:

 

復制代碼代碼如下:

use Queue;
Queue::push('CurlJsonQueue', [
 'url' => $url,
 'json' => $json
]);

 

這就是一個標準的queue壓入流程了。當然,在這里我把CurlJsonQueue類放到了services根目錄下,這個目錄已經被我注冊到composer.json的"autoload"的"classmap"中,是位于頂層命名空間中的,可以直接調用,如果需要調用非頂層命名空間,是可以寫 App/OOXX 的。我們的系統需要大量和微信服務器交互,所以就獨立出來了這個類。

 

復制代碼代碼如下:

<?php

 

class CurlJsonQueue extends BaseController{

 public function fire($job, $data)
 {
  $url = $data['url'];
  $json = $data['json'];

  parent::base_post_curl($url, $json);

  $job->delete();
 }
}

 

這個類默認的方法是 fire() ,參數也是固定的兩個 $job 和 $data,由于我在BaseController中封裝了post的curl模塊,所以就調用了一下。另外這里還有一個小坑,當時寫base_post_curl() 的時候用的protected,導致use BaseController無效,必須繼承。

通過執行上面的代碼,queue中就被放入了一個新的任務,laravel通過下面的命令開啟守護進程:

 

復制代碼代碼如下:

php artisan queue:listen

 

然后守護進程就開始處理隊列了。此代碼中的PHP命令和artisan文件的路徑請自行調整。

大家可能注意到了,我們要使用的這個隊列系統用到了redis和PHP命令行,如果在測試環境,加個開機啟動甚至是手動啟動都可以,但是在生產環境就需要更穩固的工具來守護這兩個程序,我們用的是supervisor,關于supervisor的安裝配置大家可以參考這篇文章: http://blog.segmentfault.com/qianfeng/1190000000532561 注意,文章里有小坑請自行去踩。。。

OK,全部配置好之后,跑起來redis和PHP命令行,整個系統就開始愉快地運行啦~

使用感受:

隊列服務超好用,之前一次和app的交互流程需要6-7S,異步以后降低到2S以內,基本就是傳輸時間和PHP代碼運行時間了,耗時的特殊操作已經異步了。不過隊列服務默認1S開一個進程檢查一次redis中有沒有可以運行的服務,在阿里云服務器上,大約能占到單核的10%,消耗略大,而且隊列處理時間相對較長,因為沒有了之前同步時候的文件加載福利。不過如果有多個任務,PHP進程是會連續執行的,不會1S執行一個的啦。

下面說說坑:

1. 由于queue核心類使用了一個特殊函數,導致沒有明確類型的變量會以單元素數組的形式存進json,再存進redis。解決辦法就是在每一個要放進去的數據前面加上 ''. 。上面的$url和$json由于都已經在前面用引號進行了類型申明,故沒做這一步操作。

2. 如果要傳遞url給隊列,系統queue類會在每一個 / 前面加上兩個 // 。這對于一些特殊操作可能會造成致命影響。(開玩笑,有上面那個致命么!)


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 克东县| 鹤壁市| 呼图壁县| 麻栗坡县| 彩票| 洞头县| 雷波县| 杨浦区| 民乐县| 黑山县| 汪清县| 会同县| 叙永县| 措勤县| 错那县| 鄯善县| 马龙县| 麻城市| 旺苍县| 固阳县| 犍为县| 平武县| 民勤县| 淮安市| 聂拉木县| 宜兴市| 定日县| 武威市| 吉隆县| 上林县| 巴南区| 海丰县| 梅州市| 宁乡县| 蕉岭县| 囊谦县| 武清区| 灵璧县| 交口县| 白山市| 西峡县|