這篇文章給大家分享的是有關PHP協程:Go+Chan+Defer的示例分析的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。
為永順等地區(qū)用戶提供了全套網頁設計制作服務,及永順網站建設行業(yè)解決方案。主營業(yè)務為成都網站建設、成都網站設計、永順網站設計,以傳統(tǒng)方式定制建設網站,并提供域名空間備案等一條龍服務,秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務。我們深信只要達到每一位用戶的要求,就會得到認可,從而選擇與我們長期合作。這樣,我們也可以走得更遠!
Swoole4
為PHP
語言提供了強大的CSP
協程編程模式。底層提供了3
個關鍵詞,可以方便地實現各類功能。
Swoole4
提供的PHP協程
語法借鑒自Golang
,在此向GO
開發(fā)組致敬
PHP+Swoole
協程可以與Golang
很好地互補。Golang
:靜態(tài)語言,嚴謹強大性能好,PHP+Swoole
:動態(tài)語言,靈活簡單易用
本文基于Swoole-4.2.9
和PHP-7.2.9
版本
go
:創(chuàng)建一個協程
chan
:創(chuàng)建一個通道
defer
:延遲任務,在協程退出時執(zhí)行,先進后出
這3
個功能底層實現全部為內存操作,沒有任何IO
資源消耗。就像PHP
的Array
一樣是非常廉價的。如果有需要就可以直接使用。這與socket
和file
操作不同,后者需要向操作系統(tǒng)申請端口和文件描述符,讀寫可能會產生阻塞的IO
等待。
推薦學習:《PHP視頻教程》
使用go
函數可以讓一個函數并發(fā)地去執(zhí)行。在編程過程中,如果某一段邏輯可以并發(fā)執(zhí)行,就可以將它放置到go
協程中執(zhí)行。
function test1() { sleep(1); echo "b"; } function test2() { sleep(2); echo "c"; } test1(); test2();
htf@LAPTOP-0K15EFQI:~$ time php b1.php bc real 0m3.080s user 0m0.016s sys 0m0.063s htf@LAPTOP-0K15EFQI:~$
上述代碼中,test1
和test2
會順序執(zhí)行,需要3
秒才能執(zhí)行完成。
使用go
創(chuàng)建協程,可以讓test1
和test2
兩個函數變成并發(fā)執(zhí)行。
Swoole\Runtime::enableCoroutine(); go(function () { sleep(1); echo "b"; }); go(function () { sleep(2); echo "c"; });
Swoole\Runtime::enableCoroutine()
作用是將PHP
提供的stream
、sleep
、pdo
、MySQLi
、redis
等功能從同步阻塞切換為協程的異步IO
bchtf@LAPTOP-0K15EFQI:~$ time php co.php bc real 0m2.076s user 0m0.000s sys 0m0.078s htf@LAPTOP-0K15EFQI:~$
可以看到這里只用了2
秒就執(zhí)行完成了。
順序執(zhí)行耗時等于所有任務執(zhí)行耗時的總和 :t1+t2+t3...
并發(fā)執(zhí)行耗時等于所有任務執(zhí)行耗時的最大值 :max(t1, t2, t3, ...)
有了go
關鍵詞之后,并發(fā)編程就簡單多了。與此同時又帶來了新問題,如果有2
個協程并發(fā)執(zhí)行,另外一個協程,需要依賴這兩個協程的執(zhí)行結果,如果解決此問題呢?
答案就是使用通道(Channel
),在Swoole4
協程中使用new chan
就可以創(chuàng)建一個通道。通道可以理解為自帶協程調度的隊列。它有兩個接口push
和pop
:
push
:向通道中寫入內容,如果已滿,它會進入等待狀態(tài),有空間時自動恢復
pop
:從通道中讀取內容,如果為空,它會進入等待狀態(tài),有數據時自動恢復
使用通道可以很方便地實現并發(fā)管理。
$chan = new chan(2); # 協程1 go (function () use ($chan) { $result = []; for ($i = 0; $i < 2; $i++) { $result += $chan->pop(); } var_dump($result); }); # 協程2 go(function () use ($chan) { $cli = new Swoole\Coroutine\Http\Client('www.qq.com', 80); $cli->set(['timeout' => 10]); $cli->setHeaders([ 'Host' => "www.qq.com", "User-Agent" => 'Chrome/49.0.2587.3', 'Accept' => 'text/html,application/xhtml+xml,application/xml', 'Accept-Encoding' => 'gzip', ]); $ret = $cli->get('/'); // $cli->body 響應內容過大,這里用 Http 狀態(tài)碼作為測試 $chan->push(['www.qq.com' => $cli->statusCode]); }); # 協程3 go(function () use ($chan) { $cli = new Swoole\Coroutine\Http\Client('www.163.com', 80); $cli->set(['timeout' => 10]); $cli->setHeaders([ 'Host' => "www.163.com", "User-Agent" => 'Chrome/49.0.2587.3', 'Accept' => 'text/html,application/xhtml+xml,application/xml', 'Accept-Encoding' => 'gzip', ]); $ret = $cli->get('/'); // $cli->body 響應內容過大,這里用 Http 狀態(tài)碼作為測試 $chan->push(['www.163.com' => $cli->statusCode]); });
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$ time php co2.php array(2) { ["www.qq.com"]=> int(302) ["www.163.com"]=> int(200) } real 0m0.268s user 0m0.016s sys 0m0.109s htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$
這里使用go
創(chuàng)建了3
個協程,協程2
和協程3
分別請求qq.com
和163.com
主頁。協程1
需要拿到Http
請求的結果。這里使用了chan
來實現并發(fā)管理。
協程1
循環(huán)兩次對通道進行pop
,因為隊列為空,它會進入等待狀態(tài)
協程2
和協程3
執(zhí)行完成后,會push
數據,協程1
拿到了結果,繼續(xù)向下執(zhí)行
在協程編程中,可能需要在協程退出時自動實行一些任務,做清理工作。類似于PHP
的register_shutdown_function
,在Swoole4
中可以使用defer
實現。
Swoole\Runtime::enableCoroutine(); go(function () { echo "a"; defer(function () { echo "~a"; }); echo "b"; defer(function () { echo "~b"; }); sleep(1); echo "c"; });
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$ time php defer.php abc~b~a real 0m1.068s user 0m0.016s sys 0m0.047s htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$
感謝各位的閱讀!關于“PHP協程:Go+Chan+Defer的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!