PHP的Laravel框架中使用消息隊(duì)列queue及異步隊(duì)列的方法
queue配置
首先說明一下我之前的項(xiàng)目中如何使用queue的。
我們現(xiàn)在的項(xiàng)目都是用的symfony,老一點(diǎn)的項(xiàng)目用的symfony1.4,新一點(diǎn)的項(xiàng)目用的都是symfony2。symfony用起來整體感覺還是很爽的,尤其symfony2,整體上來講使用了很多java里面框架的設(shè)計(jì)思想。但是他不支持queue。在symfony,我們使用queue也經(jīng)歷了幾個(gè)過程。最開始使用張堰同學(xué)的httpsqs。這個(gè)簡(jiǎn)單使用,但是存在單點(diǎn)。畢竟我們的項(xiàng)目還是正式對(duì)外服務(wù)的,所以我們研究了Apache旗下的開源項(xiàng)目ActiveMQ,研究研究發(fā)現(xiàn)還有Apache旗下還有更新的MQ,那就是Apollo。最后我們決定使用的Apollo。
queue在我們的項(xiàng)目中主要的應(yīng)用場(chǎng)景就是異步處理一些比較耗時(shí)的功能,比如同步第三方數(shù)據(jù)、數(shù)據(jù)有變動(dòng)了同步通知到我們的第三方數(shù)據(jù)使用者等等。我們大致的思路是這樣的,在各個(gè)controller里面如果需要異步處理的,就把一個(gè)json對(duì)象encode一下,塞到Apollo里面。再寫一個(gè)work的Command,在這個(gè)Command中解析json對(duì)象,根據(jù)里面的action和參數(shù)決定來調(diào)用不同的方法處理。根據(jù)業(yè)務(wù)需要同時(shí)在不同的機(jī)器上運(yùn)行Command作為守護(hù)進(jìn)程一直跑著,也算實(shí)現(xiàn)異步多任務(wù)處理應(yīng)用的方案。就這么一直使用著,直到發(fā)現(xiàn)了laravel。打算研究一下。如果可能替代一下也不是不可能。呵呵。
由于才開始學(xué)習(xí),當(dāng)然直接上laravel5。routes、controller、view都基本上和symfony差別不到,上手倒是不困難。最后研究一下queue。
1、安裝laravle,使用composer,倒是很簡(jiǎn)單。
composer global require "laravel/installer=~1.1" vi ~/.bash_profile
把~/.composer/vendor/bin 加入到環(huán)境變量中。
source ~/.bash_profile
就可以直接在命令行中使用laravel了。試一下。
laravel -V
能夠看到下面的,就代表成功了。
Laravel Installer version 1.2.1
2、創(chuàng)建項(xiàng)目。
laravel new guagua
3、配置redis和queue。
4、創(chuàng)建controller,
php artisan make:controller DefaultController
在controller的action中push100個(gè)queue的任務(wù)。
for($i = 0; $i < 100; $i ++) {
Queue::push(new SendEmail("ssss".$i));
}
5、創(chuàng)建queue的Command
php artisan make:command SendEmail --queued
修改app/Commands/SendEmail.php,添加一個(gè)私有變量。
protected $msg;
同時(shí)修改構(gòu)造函數(shù)。
public function __construct($msg)
{
$this->msg = $msg;
}
再修改的handle方法
public function handle() {
sleep(4);
echo $this->msg."\t".date("Y-m-d H:i:s")."\n";
$this->delete();
}
6、修改routes
Route::get('/', [
'as' => 'index',
'uses' => 'DefaultController@index'
]);
7、監(jiān)聽queue
php artisan queue:listen
為了驗(yàn)證多任務(wù)處理,我們同時(shí)開三個(gè)窗口運(yùn)行同樣的命令。
8、用laravel內(nèi)建的server啟動(dòng)服務(wù)
php artisan serve --port 8080
打開瀏覽器,訪問http://localhost:8080/頁面。當(dāng)然也可以用nginx,apache之類的。但是需要各種配置,還是內(nèi)建的使用方便。
在控制臺(tái)就能看到各個(gè)queue執(zhí)行的情況了,如下圖??梢钥吹?00個(gè)任務(wù)被三個(gè)work平分了。

到此,基本達(dá)到了我想要的效果。驗(yàn)證了laravel可以簡(jiǎn)單實(shí)現(xiàn)queue,并且可以多任務(wù)處理。
make command生成的代碼中use App\Commands\Command ,但是運(yùn)行時(shí)提示沒有這個(gè)文件。 解決辦法,修改為 use Illuminate\Console\Command; 不知道為什么會(huì)出現(xiàn)這個(gè)低級(jí)問題,難道是我mac系統(tǒng)問題,還是我的人品問題。
在controller的action中push隊(duì)列的時(shí)候,沒有異步執(zhí)行,還是在action的腳本中執(zhí)行的。 發(fā)現(xiàn)是配置問題,原來不僅僅要修改config中的queue.php,還要修改.evn中相關(guān)配置。 雖然問題解決了,但是還是覺得不能理解。還需要在學(xué)習(xí)學(xué)習(xí)laravel。
異步隊(duì)列使用方法
1.配置
關(guān)于隊(duì)列的定義,這里就不作介紹了。我們要使用異步隊(duì)列就有兩個(gè)關(guān)鍵:
(1)存儲(chǔ)隊(duì)列的地方
(2)執(zhí)行任務(wù)的服務(wù)
打開 config/queue.php ,這是Laravel5關(guān)于隊(duì)列的配置文件。首先我們可以通過 default 參數(shù)指定默認(rèn)隊(duì)列驅(qū)動(dòng),默認(rèn)配置是 sync , 這是同步隊(duì)列,我們要做異步隊(duì)列首先就要改變這里。假設(shè)我們用 database 作為驅(qū)動(dòng),隊(duì)列任務(wù)將會(huì)存放在數(shù)據(jù)庫(kù)中,而我們后面會(huì)另外啟動(dòng)一個(gè)后臺(tái)服務(wù)來處理隊(duì)列任務(wù),這就是異步方式了。
'default' => 'database'
修改完配置后,我們需要?jiǎng)?chuàng)建一個(gè)表來存放隊(duì)列任務(wù),Laravel5已經(jīng)在自帶artisan命令中內(nèi)置了一個(gè)指令用來生成數(shù)據(jù)遷移,只需要兩條命令即可,當(dāng)然你得實(shí)現(xiàn)配置好數(shù)據(jù)庫(kù)連接。
php artisan queue:table php artisan migrate
這樣就自動(dòng)在數(shù)據(jù)庫(kù)中創(chuàng)建了 jobs 表。
2.啟動(dòng)隊(duì)列監(jiān)聽服務(wù)
通過下面這條指令啟動(dòng)隊(duì)列監(jiān)聽服務(wù),它會(huì)自動(dòng)處理 jobs 表中的隊(duì)列任務(wù):
php artisan queue:listen
在linux中,如果想讓它在后臺(tái)執(zhí)行,可以這樣:
nohup php artisan queue:listen &
3.添加隊(duì)列任務(wù)
關(guān)于隊(duì)列任務(wù)的添加,手冊(cè)里說的比較詳細(xì),這里就簡(jiǎn)單舉個(gè)例子吧。
首先,通過artisan創(chuàng)建一個(gè)隊(duì)列命令:
php artisan make:command SendEmail --queued
這樣會(huì)生成 app/Commands/SendEmail.php 這個(gè)類文件,這個(gè)類會(huì)被標(biāo)識(shí)為隊(duì)列命令,你可以在 handle 方法中寫自己的業(yè)務(wù)邏輯。
在控制器中,可以簡(jiǎn)單通過 Bus::dispatch 分發(fā)任務(wù):
Bus::dispatch(new \App\Commands\SendEmail());
你會(huì)發(fā)現(xiàn)任務(wù)不會(huì)立即執(zhí)行,而是被放到 jobs 表中,由隊(duì)列監(jiān)聽服務(wù)處理。
更詳細(xì)的用法建議參考 command bus 和 queue 相關(guān)的手冊(cè)章節(jié)。
相關(guān)文章
實(shí)例講解yii2.0在php命令行中運(yùn)行的步驟
Yii中的資源是和Web頁面相關(guān)的文件,可為CSS文件,JavaScript文件,圖片或視頻等,資源放在Web可訪問的目錄下,直接被Web服務(wù)器調(diào)用。本文通過實(shí)例講解yii2.0在php命令行中運(yùn)行的步驟,對(duì)yii2.0 php相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)吧2015-12-12
Laravel執(zhí)行migrate命令提示:No such file or directory的解決方法
這篇文章主要介紹了Laravel執(zhí)行migrate命令提示:No such file or directory的解決方法,分析了執(zhí)行migrate命令出現(xiàn)錯(cuò)誤的原因與相關(guān)的解決方法,需要的朋友可以參考下2016-03-03
phpinfo 系統(tǒng)查看參數(shù)函數(shù)代碼
并根據(jù)自身的理解做了很多修改和優(yōu)化,就當(dāng)前而言,這是探測(cè)信息最全面的PHP探針了!2009-06-06
Thinkphp 在api開發(fā)中異常返回依然是html的解決方式
今天小編就為大家整理了一篇Thinkphp 在api開發(fā)中異常返回依然是html的解決方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-10-10
PHP實(shí)現(xiàn)微信小程序在線支付功能(代碼實(shí)例)
這篇文章主要介紹了PHP微信小程序在線支付功能(代碼實(shí)例),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的工作或?qū)W習(xí)具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
PHP stream_context_create()函數(shù)的使用示例
這篇文章主要介紹了PHP stream_context_create()函數(shù)的使用示例,stream_context_create()函數(shù)是用來 創(chuàng)建打開文件的上下文件選項(xiàng),用于fopen(),file_get_contents()等過程的超時(shí)設(shè)置、代理服務(wù)器、請(qǐng)求方式、頭信息設(shè)置的特殊過程,需要的朋友可以參考下2015-05-05

