首頁 >後端開發 >php教程 >詳解PHP協程:Go + Chan + Defer

詳解PHP協程:Go + Chan + Defer

藏色散人
藏色散人轉載
2021-06-27 17:01:184180瀏覽

#Swoole4PHP語言提供了強大的CSP協程程式模式。底層提供了3個關鍵字,可以方便實現各類功能。

  • Swoole4提供的PHP協程語法借鑒自Golang,在此向GO開發組致敬
  • PHP Swoole協程可以與Golang很好地互補。 Golang:靜態語言,嚴謹強大性能好,PHP Swoole:動態語言,靈活簡單易用
本文基於Swoole-4.2. 9PHP-7.2.9版本

關鍵字

  • go :建立一個協程
  • chan :建立一個通道
  • defer :延遲任務,在協程退出時執行,先進後出

3個功能底層實作全部為記憶體運算,沒有任何IO資源消耗。就像PHPArray一樣是非常廉價的。如果有需要就可以直接使用。這與socketfile操作不同,後者需要向作業系統申請連接埠和檔案描述符,讀寫可能會產生阻塞的IO等待。

推薦學習:《PHP影片教學

協程並發

使用go函數可以讓一個函數並發地去執行。在程式設計過程中,如果某一段邏輯可以並發執行,就可以將它放置到go協程中執行。

順序執行

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:~$

上述程式碼中,test1test2會順序執行,需要3秒才能執行完成。

並發執行

使用go建立協程,可以讓test1test2兩個函數變成並發執行。

Swoole\Runtime::enableCoroutine();

go(function () 
{
    sleep(1);
    echo "b";
});
    
go(function () 
{
    sleep(2);
    echo "c";
});
Swoole\Runtime::enableCoroutine()作用是將PHP提供的streamsleeppdomysqliredis等功能從同步阻塞切換為協程的非同步IO
##執行結果:

bchtf@LAPTOP-0K15EFQI:~$ time php co.php
bc
real    0m2.076s
user    0m0.000s
sys     0m0.078s
htf@LAPTOP-0K15EFQI:~$

可以看到這裡只花了

2秒就執行完成了。

    順序執行耗時等於所有任務執行耗時的總和:
  • t1 t2 t3...
  • 並發執行耗時等於所有任務執行耗時的最大值:
  • max(t1, t2, t3, ...)
#協程通訊

有了

go#關鍵詞之後,並發程式就簡單多了。同時又帶來了新問題,如果有2個協程並發執行,另外一個協程,需要依賴這兩個協程的執行結果,如果解決此問題呢?

答案就是要使用頻道(

Channel),在Swoole4協程中使用new chan就可以建立一個頻道。通道可以理解為自帶協程調度的隊列。它有兩個介面pushpop

  • #push:寫入通道中寫入內容,如果已滿,它會進入等待狀態,有空間時自動恢復
  • pop:從通道中讀取內容,如果為空,它會進入等待狀態,有資料時會自動恢復
使用通道可以很方便地實作

並發管理

$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 状态码作为测试
   $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 状态码作为测试
   $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建立了3個協程,協程2和協程3分別請求qq.com163.com首頁。協程1需要拿到Http請求的結果。這裡使用了chan來實現並發管理。

    協程
  • 1循環兩次對通道進行pop,因為佇列為空,它會進入等待狀態
  • 協程
  • 2和協程3執行完成後,會push數據,協程1拿到了結果,繼續向下執行
延遲任務

在協程程式設計中,可能需要在協程退出時自動實行一些任務,做清理工作。類似於

PHPregister_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$

結語

#Swoole4提供的Go Chan DeferPHP帶來了一種全新的CSP並發程式設計模式。靈活使用Swoole4提供的各項特性,可解決工作中各類複雜功能的設計與開發。

以上是詳解PHP協程:Go + Chan + Defer的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:segmentfault.com。如有侵權,請聯絡admin@php.cn刪除