背景##
正在做一個智能家居的項目,接收下位機(就是控制智能家居硬件模塊的HUB)協(xié)議解析,Web端維護硬件狀態(tài),利用APP交互。由于下位機數(shù)據(jù)是發(fā)送到服務(wù)器的XXX端口,所以必須對XXX端口進行監(jiān)聽。其實和聊天室的概念差不多,研究了一下workerman、swoole和其他幾個開源的項目,決定采用swoole。
關(guān)于php解析下位機的16進制協(xié)議,其實相當之扯蛋,要是你最好還是用.NET或者JAVA吧。很久沒碰MVC了,所以直接上PHP吧。網(wǎng)上搜搜還沒見幾個php這樣搞的項目,我還沒做完,做完了來談?wù)?,關(guān)鍵函數(shù)主要是bin2hex/pack/unpack。這一篇主要聊聊Laravel如何優(yōu)雅的使用Swoole,其實只需簡單3步就可以完成。
什么是Swoole##
直接套用Swoole官網(wǎng)的介紹:PHP的異步、并行、高性能網(wǎng)絡(luò)通信引擎,使用純C語言編寫,提供了PHP語言的異步多線程服務(wù)器,異步TCP/UDP網(wǎng)絡(luò)客戶端,異步MySQL,異步Redis,數(shù)據(jù)庫連接池,AsyncTask,消息隊列,毫秒定時器,異步文件讀寫,異步DNS查詢。 Swoole內(nèi)置了Http/WebSocket服務(wù)器端/客戶端、Http2.0服務(wù)器端。
Swoole官網(wǎng)的文檔不夠豐富啊,這比較頭疼,但大部分的問題都解釋了。如果你對Swoole很感興趣,那么看看這個Swoole入門教程。Swoole提供了多線程、長連接等很多牛逼的功能,把php上升到了一個新的臺階,具體的你可以看看入門教程,本文只限于討論Laravel和Swoole的結(jié)合。
Swoole為了提供服務(wù),必須以CLI模式運行,什么是CLI模式呢?如果你Swoole業(yè)務(wù)代碼是寫在一個叫server.php的文件中,那么在命令行下輸入php server.php開啟。這是比較頭疼的事情,因為Laravel框架可不是這樣的運轉(zhuǎn)的,那如何能與Laravel結(jié)合呢?沒錯,自定義一條Artisan Command,就這么簡單。
STEP 1-自定義Command##
關(guān)于自定義Artisan Commnad,你需要了解的技術(shù)點都在這里,我自定義了一個叫做SwooleCommand的命令,直接貼關(guān)鍵代碼:

在命令行(CLI)下執(zhí)行php artisan swoole start即可開啟Swoole服務(wù)。分析一下代碼,你可以看到命令參數(shù)包括啟動、重啟、關(guān)閉,我圖省事只實現(xiàn)了啟動部分,如果需要關(guān)閉,在linux中利用kill命令關(guān)閉進程,步驟挺簡單的:
1.執(zhí)行 ps -aux|grep artisan命令,獲取pid(有多個進程,殺第一個即可)
2.執(zhí)行 kill pid命令,pid是第一步你獲取的
3.如果想后臺值守,一定加上nohup命令?。?!
關(guān)于Swoole的配置不是本文討論的范圍,請移步官網(wǎng),這里把Swoole服務(wù)用$serv變量進行了保存,是為了后面Laravel發(fā)送命令交互。你可以看到,Swoole的事件響應(yīng)代碼是這樣的:

如果說fire打開了Swoole的大門,那么這里的handler就是Swoole與Laravel的傳送帶,利用自己寫的handler,就可以把各種業(yè)務(wù)邏輯寫進Laravel框架中,然后就可以使用Laravel提供的各種高效方便的功能了。“handler”是一種命名習慣,你也可以叫做"callback"、"manager"、"listener",這看你的命名習慣了。我沒有采用new的方式而是用Laravel的IoC注入App::make,主要是圖省事(因為handler的構(gòu)造器用到了我自定義的數(shù)據(jù)處理類,往下看)。
STEP 2-自定義handler##
因為是自定義的類,請遵循命名空間,并在composer.json中聲明,完了執(zhí)行composer dump-autoload命令更新一遍。比如我創(chuàng)建了一個文件夾app\handlers存放handler,那么在composer.json中看起來是這樣的:

那么handler里面具體干些啥,就由你來決定了。反正和寫controller差不多,各種Laravel框架的功能你都能隨便用,貼上我的:

上一節(jié)我提到我用IoC是因為構(gòu)造器里面用到了自己的數(shù)據(jù)處理類,我把增刪改查和其他數(shù)據(jù)處理的業(yè)務(wù)放到Repository中了,沒其他原因,只是這樣代碼看起來清爽一點。如此,利用Swoole接收數(shù)據(jù)的流程就算搞定了,那么要想利用Swoole向客戶端發(fā)送數(shù)據(jù)該怎么做呢?咳咳,這個稍微麻煩點,需要曲線方法實現(xiàn),繼續(xù)看下一節(jié)。
STEP 3-發(fā)送數(shù)據(jù)##
有兩種方法,但都離不開一個緩存kv結(jié)構(gòu)(Laravel自帶的Cache功能就夠了),保存客戶端的地址數(shù)據(jù),要不你怎么知道發(fā)到哪里去。我用的是第一種,圖省事,發(fā)送數(shù)據(jù)和Swoole就無關(guān)了,如果你需要長連接websocket,這種不適用,老老實實用第二種吧。如果你有更好的辦法,請一定要告訴我!
第一種:fsockopen####
挺簡單的,和swoole就沒關(guān)系了,利用Swoole的connection_info函數(shù)獲取客戶端的IP地址和端口,然后用fsockopen直接發(fā)送數(shù)據(jù)。
第二種:內(nèi)部端口監(jiān)聽####
Swoole支持監(jiān)聽多個端口,實現(xiàn)的思想就是利用fsockopen把數(shù)據(jù)利用內(nèi)部監(jiān)聽的端口發(fā)送過去,然后就可以調(diào)用$serv發(fā)送消息了。這么做的好處就是不需要知道客戶端的實際IP地址和端口,在Cache保存客戶端的$fd標識,直接就發(fā)數(shù)據(jù)。采用這個思路,請記得iptables把端口打開。我自己并沒有采用,因為不是長連接我覺得太麻煩。
總結(jié)##
Swoole非常棒,其實都沒怎么用上。你還可以參考官網(wǎng)的配置,將Swoole作為nginx承載代理,據(jù)說性能提升很大。