摘要:為語言提供了強(qiáng)大的協(xié)程編程模式。提供的協(xié)程語法借鑒自,在此向開發(fā)組致敬協(xié)程可以與很好地互補(bǔ)。并發(fā)執(zhí)行使用創(chuàng)建協(xié)程,可以讓和兩個(gè)函數(shù)變成并發(fā)執(zhí)行。協(xié)程需要拿到請(qǐng)求的結(jié)果。
Swoole4為PHP語言提供了強(qiáng)大的CSP協(xié)程編程模式。底層提供了3個(gè)關(guān)鍵詞,可以方便地實(shí)現(xiàn)各類功能。
Swoole4提供的PHP協(xié)程語法借鑒自Golang,在此向GO開發(fā)組致敬
PHP+Swoole協(xié)程可以與Golang很好地互補(bǔ)。Golang:靜態(tài)語言,嚴(yán)謹(jǐn)強(qiáng)大性能好,PHP+Swoole:動(dòng)態(tài)語言,靈活簡單易用
本文基于Swoole-4.2.9和PHP-7.2.9版本關(guān)鍵詞
go :創(chuàng)建一個(gè)協(xié)程
chan :創(chuàng)建一個(gè)通道
defer :延遲任務(wù),在協(xié)程退出時(shí)執(zhí)行,先進(jìn)后出
這3個(gè)功能底層實(shí)現(xiàn)全部為內(nèi)存操作,沒有任何IO資源消耗。就像PHP的Array一樣是非常廉價(jià)的。如果有需要就可以直接使用。這與socket和file操作不同,后者需要向操作系統(tǒng)申請(qǐng)端口和文件描述符,讀寫可能會(huì)產(chǎn)生阻塞的IO等待。
協(xié)程并發(fā)使用go函數(shù)可以讓一個(gè)函數(shù)并發(fā)地去執(zhí)行。在編程過程中,如果某一段邏輯可以并發(fā)執(zhí)行,就可以將它放置到go協(xié)程中執(zhí)行。
順序執(zhí)行function test1() { sleep(1); echo "b"; } function test2() { sleep(2); echo "c"; } test1(); test2();執(zhí)行結(jié)果:
htf@LAPTOP-0K15EFQI:~$ time php b1.php bc real 0m3.080s user 0m0.016s sys 0m0.063s htf@LAPTOP-0K15EFQI:~$
上述代碼中,test1和test2會(huì)順序執(zhí)行,需要3秒才能執(zhí)行完成。
并發(fā)執(zhí)行使用go創(chuàng)建協(xié)程,可以讓test1和test2兩個(gè)函數(shù)變成并發(fā)執(zhí)行。
SwooleRuntime::enableCoroutine(); go(function () { sleep(1); echo "b"; }); go(function () { sleep(2); echo "c"; });
SwooleRuntime::enableCoroutine()作用是將PHP提供的stream、sleep、pdo、mysqli、redis等功能從同步阻塞切換為協(xié)程的異步IO執(zhí)行結(jié)果:
bchtf@LAPTOP-0K15EFQI:~$ time php co.php bc real 0m2.076s user 0m0.000s sys 0m0.078s htf@LAPTOP-0K15EFQI:~$
可以看到這里只用了2秒就執(zhí)行完成了。
順序執(zhí)行耗時(shí)等于所有任務(wù)執(zhí)行耗時(shí)的總和 :t1+t2+t3...
并發(fā)執(zhí)行耗時(shí)等于所有任務(wù)執(zhí)行耗時(shí)的最大值 :max(t1, t2, t3, ...)
協(xié)程通信有了go關(guān)鍵詞之后,并發(fā)編程就簡單多了。與此同時(shí)又帶來了新問題,如果有2個(gè)協(xié)程并發(fā)執(zhí)行,另外一個(gè)協(xié)程,需要依賴這兩個(gè)協(xié)程的執(zhí)行結(jié)果,如果解決此問題呢?
答案就是使用通道(Channel),在Swoole4協(xié)程中使用new chan就可以創(chuàng)建一個(gè)通道。通道可以理解為自帶協(xié)程調(diào)度的隊(duì)列。它有兩個(gè)接口push和pop:
push:向通道中寫入內(nèi)容,如果已滿,它會(huì)進(jìn)入等待狀態(tài),有空間時(shí)自動(dòng)恢復(fù)
pop:從通道中讀取內(nèi)容,如果為空,它會(huì)進(jìn)入等待狀態(tài),有數(shù)據(jù)時(shí)自動(dòng)恢復(fù)
使用通道可以很方便地實(shí)現(xiàn)并發(fā)管理。
$chan = new chan(2); # 協(xié)程1 go (function () use ($chan) { $result = []; for ($i = 0; $i < 2; $i++) { $result += $chan->pop(); } var_dump($result); }); # 協(xié)程2 go(function () use ($chan) { $cli = new SwooleCoroutineHttpClient("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 響應(yīng)內(nèi)容過大,這里用 Http 狀態(tài)碼作為測試 $chan->push(["www.qq.com" => $cli->statusCode]); }); # 協(xié)程3 go(function () use ($chan) { $cli = new SwooleCoroutineHttpClient("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 響應(yīng)內(nèi)容過大,這里用 Http 狀態(tài)碼作為測試 $chan->push(["www.163.com" => $cli->statusCode]); });執(zhí)行結(jié)果:
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個(gè)協(xié)程,協(xié)程2和協(xié)程3分別請(qǐng)求qq.com和163.com主頁。協(xié)程1需要拿到Http請(qǐng)求的結(jié)果。這里使用了chan來實(shí)現(xiàn)并發(fā)管理。
協(xié)程1循環(huán)兩次對(duì)通道進(jìn)行pop,因?yàn)殛?duì)列為空,它會(huì)進(jìn)入等待狀態(tài)
協(xié)程2和協(xié)程3執(zhí)行完成后,會(huì)push數(shù)據(jù),協(xié)程1拿到了結(jié)果,繼續(xù)向下執(zhí)行
延遲任務(wù)在協(xié)程編程中,可能需要在協(xié)程退出時(shí)自動(dòng)實(shí)行一些任務(wù),做清理工作。類似于PHP的register_shutdown_function,在Swoole4中可以使用defer實(shí)現(xiàn)。
SwooleRuntime::enableCoroutine(); go(function () { echo "a"; defer(function () { echo "~a"; }); echo "b"; defer(function () { echo "~b"; }); sleep(1); echo "c"; });執(zhí)行結(jié)果:
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$結(jié)語
Swoole4提供的Go + Chan + Defer為PHP帶來了一種全新的CSP并發(fā)編程模式。靈活使用Swoole4提供的各項(xiàng)特性,可以解決工作中各類復(fù)雜功能的設(shè)計(jì)和開發(fā)。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/29741.html
摘要:的異步并行高性能網(wǎng)絡(luò)通信引擎已發(fā)布版本。新版本提供了全新的短名,完整支持了協(xié)程通道特性,為語言帶來了全新的編程模式。的借鑒至語言,在此向語言開發(fā)組致敬。其他服務(wù)器端提供了很多共個(gè)支持協(xié)程的類,可以在這些服務(wù)器程序中使用協(xié)程。 PHP的異步、并行、高性能網(wǎng)絡(luò)通信引擎 Swoole 已發(fā)布 2.1.0 版本。新版本提供了全新的短名 API,完整支持了協(xié)程(Coroutine)+通道(Cha...
摘要:官方文檔中對(duì)協(xié)程的示例大多按照一次請(qǐng)求一個(gè)協(xié)程或腳本并發(fā)大量協(xié)程的方式來舉例說明這種使用方式下提升的是整體的性能而非單次請(qǐng)求的響應(yīng)時(shí)間要提升單次請(qǐng)求的響應(yīng)效率或提升非網(wǎng)絡(luò)服務(wù)下腳本代碼的運(yùn)行效率需要在業(yè)務(wù)代碼中主動(dòng)使用協(xié)程來處理那些可并發(fā)的 Swoole官方文檔中對(duì)協(xié)程的示例大多按照一次請(qǐng)求一個(gè)協(xié)程(或腳本并發(fā)大量協(xié)程)的方式來舉例說明,這種使用方式下提升的是整體的性能,而非單次請(qǐng)求的...
摘要:當(dāng)協(xié)程執(zhí)行權(quán)讓渡回來的時(shí)候,把原來的上下文恢復(fù)。說明協(xié)程是并發(fā)的。實(shí)際的收益取決于后端的服務(wù)的延遲,如果耗時(shí)很長,通過協(xié)程并發(fā)則可以收益明顯。 想法很簡單。通過設(shè)置 runtime.GOMAXPROCS(1) 讓 golang 的進(jìn)程變成單線程執(zhí)行的。類似python用gevent的效果。然后通過調(diào)度多個(gè)協(xié)程實(shí)現(xiàn)異步I/O并發(fā)。php作為一個(gè)子函數(shù)跑在go的進(jìn)程內(nèi),php需要yield...
閱讀 3986·2021-11-23 10:09
閱讀 1347·2021-11-23 09:51
閱讀 2946·2021-11-23 09:51
閱讀 1595·2021-09-07 09:59
閱讀 2359·2019-08-30 15:55
閱讀 2306·2019-08-30 15:55
閱讀 2955·2019-08-30 15:52
閱讀 2568·2019-08-26 17:04