溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

PHP協(xié)程:Go+Chan+Defer的示例分析

發(fā)布時(shí)間:2021-06-28 11:34:38 來(lái)源:億速云 閱讀:195 作者:小新 欄目:編程語(yǔ)言

這篇文章給大家分享的是有關(guān)PHP協(xié)程:Go+Chan+Defer的示例分析的內(nèi)容。小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。

Swoole4PHP語(yǔ)言提供了強(qiáng)大的CSP協(xié)程編程模式。底層提供了3個(gè)關(guān)鍵詞,可以方便地實(shí)現(xiàn)各類功能。

  • Swoole4提供的PHP協(xié)程語(yǔ)法借鑒自Golang,在此向GO開(kāi)發(fā)組致敬

  • PHP+Swoole協(xié)程可以與Golang很好地互補(bǔ)。Golang:靜態(tài)語(yǔ)言,嚴(yán)謹(jǐn)強(qiáng)大性能好,PHP+Swoole:動(dòng)態(tài)語(yǔ)言,靈活簡(jiǎn)單易用

本文基于Swoole-4.2.9PHP-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)存操作,沒(méi)有任何IO資源消耗。就像PHPArray一樣是非常廉價(jià)的。如果有需要就可以直接使用。這與socketfile操作不同,后者需要向操作系統(tǒng)申請(qǐng)端口和文件描述符,讀寫可能會(huì)產(chǎn)生阻塞的IO等待。

推薦學(xué)習(xí):《PHP視頻教程》

協(xié)程并發(fā)

使用go函數(shù)可以讓一個(gè)函數(shù)并發(fā)地去執(zhí)行。在編程過(guò)程中,如果某一段邏輯可以并發(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:~$

上述代碼中,test1test2會(huì)順序執(zhí)行,需要3秒才能執(zhí)行完成。

并發(fā)執(zhí)行

使用go創(chuàng)建協(xié)程,可以讓test1test2兩個(gè)函數(shù)變成并發(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等功能從同步阻塞切換為協(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ā)編程就簡(jiǎn)單多了。與此同時(shí)又帶來(lái)了新問(wèn)題,如果有2個(gè)協(xié)程并發(fā)執(zhí)行,另外一個(gè)協(xié)程,需要依賴這兩個(gè)協(xié)程的執(zhí)行結(jié)果,如果解決此問(wèn)題呢?

答案就是使用通道(Channel),在Swoole4協(xié)程中使用new chan就可以創(chuàng)建一個(gè)通道。通道可以理解為自帶協(xié)程調(diào)度的隊(duì)列。它有兩個(gè)接口pushpop

  • 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 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 響應(yīng)內(nèi)容過(guò)大,這里用 Http 狀態(tài)碼作為測(cè)試
   $chan->push(['www.qq.com' => $cli->statusCode]);
});

# 協(xié)程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 響應(yīng)內(nèi)容過(guò)大,這里用 Http 狀態(tài)碼作為測(cè)試
   $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.com163.com主頁(yè)。協(xié)程1需要拿到Http請(qǐng)求的結(jié)果。這里使用了chan來(lái)實(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ù),做清理工作。類似于PHPregister_shutdown_function,在Swoole4中可以使用defer實(shí)現(xiàn)。

Swoole\Runtime::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$

感謝各位的閱讀!關(guān)于“PHP協(xié)程:Go+Chan+Defer的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI