php +go關鍵字實現協程


來源: https://studygolang.com/articles/17631?fr=sidebar

 

今天在知乎瀏覽時忽然發現了一個有趣的東西,php竟然可以實現協程的實現,而且還是通過go關鍵字實現,頓時感覺php現在發展的好迅速,竟然把go里的東西都借鑒去。
只不過這是在一個叫Swoole的框架中實現的。
Swoole4為PHP語言提供了強大的CSP協程編程模式。底層提供了3個關鍵詞,可以方便地實現各類功能。
Swoole4提供的PHP協程語法借鑒自Golang,在此向GO開發組致敬
PHP+Swoole協程可以與Golang很好地互補。Golang:靜態語言,嚴謹強大性能好,PHP+Swoole:動態語言,靈活簡單易用

 

關鍵詞

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

 

這3個功能底層實現全部為內存操作,沒有任何IO資源消耗。就像PHP的Array一樣是非常廉價的。如果有需要就可以直接使用。這與socket和file操作不同,后者需要向操作系統申請端口和文件描述符,讀寫可能會產生阻塞的IO等待。

協程並發

function test1() 
{
    sleep(1);
    echo "b";
}
    
function test2() 
{
    sleep(2);
    echo "c";
}

test1();
test2();

 

執行結果:

bc
real    0m3.080s
user    0m0.016s
sys     0m0.063s
htf@LAPTOP-0K15EFQI:~$

上述代碼中,test1和test2會順序執行,需要3秒才能執行完成。

 

並發執行

使用go創建協程,可以讓test1和test2兩個函數變成並發執行。

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

 

協程通信

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

答案就是使用通道(Channel),在Swoole4協程中使用new chan就可以創建一個通道。通道可以理解為自帶協程調度的隊列。它有兩個接口push和pop:

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]);
});

 

這里使用go創建了3個協程,協程2和協程3分別請求qq.com和163.com主頁。協程1需要拿到Http請求的結果。這里使用了chan來實現並發管理。

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

Swoole4提供的Go + Chan + Defer為PHP帶來了一種全新的CSP並發編程模式。靈活使用Swoole4提供的各項特性,可以解決工作中各類復雜功能的設計和開發。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM