摘要:生成器委托簡單地翻譯官方文檔的描述中,通過生成器委托,可以將其他生成器可迭代的對象數組委托給外層生成器。外層的生成器會先順序委托出來的值,然后繼續本身中定義的值。文章介紹了的迭代生成器,協程,并實現了一個簡單的非阻塞服務器。
生成器委托
簡單地翻譯官方文檔的描述:
PHP7中,通過生成器委托(yield from),可以將其他生成器、可迭代的對象、數組委托給外層生成器。外層的生成器會先順序 yield 委托出來的值,然后繼續 yield 本身中定義的值。
利用 yield from 可以方便我們編寫比較清晰生成器嵌套,而代碼嵌套調用是編寫復雜系統所必需的。
上例子:
以上將輸出:
foo iteration 1 foo iteration 2 foo iteration 3 foo iteration 4 foo iteration 5 foo iteration 6 foo iteration 7 foo iteration 8 foo iteration 9 foo iteration 10 --- bar iteration 1 bar iteration 2 bar iteration 3 bar iteration 4 bar iteration 5自然,內部生成器也可以接受它的父生成器發送的信息或者異常,因為 yield from 為父子生成器建立一個雙向的通道。不多說,上例子:
send($num); } $gen->send(null); foreach (range(1,5) as $num) { $gen->send($num); } //$gen->send("hello world"); //try it ,gay輸出和上個例子是一樣的。
生成器返回值如果生成器被迭代完成,或者運行到 return 關鍵字,是會給這個生成器返回值的。
可以有兩種方法獲取這個返回值:使用 $ret = Generator::getReturn() 方法。
使用 $ret = yield from Generator() 表達式。
上例子:
getReturn(); } foreach (task() as $item) { ; }輸出結果就不貼了,想必大家都猜到。
可以看到 yield from 和 return 結合使得 yield 的寫法更像平時我們寫的同步模式的代碼了,畢竟,這就是 PHP 出生成器特性的原因之一呀。
一個非阻塞的web服務器時間回到2015年,鳥哥博客上轉載的一篇《 在PHP中使用協程實現多任務調度》。文章介紹了PHP5 的迭代生成器,協程,并實現了一個簡單的非阻塞 web 服務器。(鏈接見文末引用)
現在我們利用 PHP7 中的這兩個新特性重寫這個 web 服務器,只需要 100 多行代碼。
代碼如下:
socket = $socket; $this->masterCoSocket = $master ?? $this; } public function accept() { $isSelect = yield from $this->onRead(); $acceptS = null; if ($isSelect && $as = stream_socket_accept($this->socket, 0)) { $acceptS = new CoSocket($as, $this); } return $acceptS; } public function read($size) { yield from $this->onRead(); yield ($data = fread($this->socket, $size)); return $data; } public function write($string) { yield from $this->onWriter(); yield fwrite($this->socket, $string); } public function close() { unset($this->masterCoSocket->streamPoolRead[(int)$this->socket]); unset($this->masterCoSocket->streamPoolWrite[(int)$this->socket]); yield ($success = @fclose($this->socket)); return $success; } public function onRead($timeout = null) { $this->masterCoSocket->streamPoolRead[(int)$this->socket] = $this->socket; $pool = $this->masterCoSocket->streamPoolRead; $rSocks = []; $wSocks = $eSocks = null; foreach ($pool as $item) { $rSocks[] = $item; } yield ($num = stream_select($rSocks, $wSocks, $eSocks, $timeout)); return $num; } public function onWriter($timeout = null) { $this->masterCoSocket->streamPoolWrite[(int)$this->socket] = $this->socket; $pool = $this->masterCoSocket->streamPoolRead; $wSocks = []; $rSocks = $eSocks = null; foreach ($pool as $item) { $wSocks[] = $item; } yield ($num = stream_select($rSocks, $wSocks, $eSocks, $timeout)); return $num; } public function onRequest() { /** @var self $socket */ $socket = yield from $this->accept(); if (empty($socket)) { return false; } $data = yield from $socket->read(8192); $response = call_user_func($this->handleCallback, $data); yield from $socket->write($response); return yield from $socket->close(); } public static function start($port, callable $callback) { echo "Starting server at port $port... "; $socket = @stream_socket_server("tcp://0.0.0.0:$port", $errNo, $errStr); if (!$socket) throw new Exception($errStr, $errNo); stream_set_blocking($socket, 0); $coSocket = new self($socket); $coSocket->handleCallback = $callback; function gen($coSocket) { /** @var self $coSocket */ while (true) yield from $coSocket->onRequest(); } foreach (gen($coSocket) as $item){}; } } CoSocket::start(8000, function ($data) { $response = <<參考資料 [1] http://www.php.net/manual/zh/...
[2] http://www.laruence.com/2015/...
[3] http://blog.csdn.net/u0101613...
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/25582.html
摘要:以下分別備忘版本的部分新增特性。而且,在生成器沒有輸出完全時,使用會報錯。報錯定義常量數組多聲明數組解構運行時將部分指令轉為機器碼。對于計算密集型應用有較高的性能提升。 以下分別備忘 PHP 7.0 & 7.1 版本的部分新增特性。 PHP 7.0 ?? 運算符 $foo = null; $bar = $foo ?? 123; // 相當于 $bar = isset($bar) ? ...
python中的生成器、迭代器、裝飾器分別是什么意思呢?具體的含義,一些其具體的用途,下面小編就給大家詳細的解答下。 一、裝飾器 由于一個函數能實現一種功能,現在想要在不改變其代碼的情況下,讓這個函數進化一下,即能保持原來的功能,還能有新的"技能",怎么辦? 現已經存在一個自定義的函數func1 deffunc1(): print('hello,worl...
摘要:最近項目中用的比較多,所以特地去了解,模仿一下實現先來看看使用的方法通過是通過使用生成器配合方法模擬的一個同步操作,這個技術有效的避免了傳統回調和形成的回調地獄。 最近項目中 asyn & await 用的比較多,所以特地去了解,模仿一下實現~ 先來看看 使用 async & await 的方法 async function d () { try { const a = a...
閱讀 2032·2021-11-08 13:14
閱讀 2940·2021-10-18 13:34
閱讀 2029·2021-09-23 11:21
閱讀 3591·2019-08-30 15:54
閱讀 1760·2019-08-30 15:54
閱讀 2931·2019-08-29 15:33
閱讀 2581·2019-08-29 14:01
閱讀 1948·2019-08-29 13:52