# http://php.net/manual/zh/class.generator.php Generator implements Iterator { /* Methods */ //獲取迭代器當前值 public mixed current ( void ) //獲取迭代器當前值 public mixed getReturn ( void ) //返回當前產生的鍵 public mixed key ( void ) //生成器從上一次yield處繼續執行 public void next ( void ) //重置迭代器 public void rewind ( void ) //向生成器中傳入一個值 public mixed send ( mixed $value ) //向生成器中拋入一個異常 public mixed throw ( Throwable $exception ) //檢查迭代器是否被關閉 public bool valid ( void ) //迭代器序列化時執行的方法 public void __wakeup ( void ) }
$gen = new Generator(); # 咱們發現不能直接手動實例化 # output PHP Fatal error: Uncaught Error: The "Generator" class is reserved for internal use and cannot be manually instantiated in /web/www/sxx_admin3/src/cache/test/amphp/gen3.php:8
function gen($max) { for ($i=0; $i<$max; $i++) { yield $i; } } $gen = gen(5); # success # 成功,咱們只須要在普通函數方法裏yield便可成了生成器
其實各語言都有生成器,好比python,go等php
被代碼將演示valid, getReturnpython
function gen($max) { for ($i=0; $i<$max; $i++) { yield $i; } return $max; } $gen = gen(5); foreach ($gen as $val) { var_dump($val); } //若是已經迭代完成,獲取返回值 // php7 支持 // valid 判斷當前迭代器是否迭代完成 // getReturn 返回迭代器的返回值 if (version_compare(PHP_VERSION, '7.0.0') >= 0 && !$gen->valid()) { var_dump($gen->getReturn()); }
迭代器返回值能夠帶key和value,相似web
function gen($max) { for ($i=0; $i<$max; $i++) { yield $i => $i+1; } return $max; } $gen = gen(5); //var_dump($gen->key()); //var_dump($gen->current()); foreach ($gen as $key=>$val) { var_dump($key . "=>" . $val); } # output string(4) "0=>1" string(4) "1=>2" string(4) "2=>3" string(4) "3=>4" string(4) "4=>5"
本代碼將演示rewind, next, send方法segmentfault
function gen($max) { for ($i=0; $i<$max; $i++) { // 此處的(yield $i)在php7之後版本可省略 $res = (yield $i); var_dump($res); } return $max; } $gen = gen(10); // 可不調用,隱式調用 // 若是迭代開始後不能再rewind(即便用了next或send後) $gen->rewind(); // 打印獲取到當前生成器的值 var_dump("1::" . $gen->current()); //output: string(4) "1::0" // 下面2句代碼執行,將返回錯誤 // $gen->next(); // $gen->rewind(); //繼續執行,知道遇到下一個yield $gen->next(); var_dump("2::" . $gen->current()); //output: string(4) "2::1" $gen->next(); var_dump("3::" . $gen->current()); //output: string(4) "3::2" // send傳null值等同於調用next(本方法嘗試來自python的迭代器,成功) $gen->send(null); var_dump("4::" . $gen->current()); //output: string(4) "4::3" // send傳值會也會繼續執行 $gen->send(100); var_dump("5::" . $gen->current()); //output: string(4) "5::4" //若是已經迭代完成,獲取返回值 // php7 支持 if (version_compare(PHP_VERSION, '7.0.0') >= 0 && !$gen->valid()) { var_dump($gen->getReturn()); } # output: string(4) "1::0" NULL string(4) "2::1" NULL string(4) "3::2" NULL string(4) "4::3" int(100) string(4) "5::4" # 咱們先不去理會gen裏var_dump輸出的NULL或int(100) # 咱們先去理解每次next後current能夠獲取到當前yield的值便可
function gen($max) { for ($i=0; $i<$max; $i++) { $res = (yield $i); var_dump($res); } return $max; } $gen = gen(10); var_dump("1::" . $gen->current()); $gen->send(222); var_dump("2::" . $gen->current()); $gen->send(333); var_dump("3::" . $gen->current()); $gen->send(null); var_dump("4::" . $gen->current()); # output: string(4) "1::0" int(222) string(4) "2::1" int(333) string(4) "3::2" int(444) string(4) "4::3" # send和next # next() => current = yield值 # send(val) $rs = yield 表達式執行 = val; //send這樣理解便可 # 在當前某個yield處時send,當前yield表達式處返回,若是沒有變量接收,那麼繼續下一個yield處返回 $rs = (yield somethind_to_do(...) ); ^ |-------------------| | yield值 | |----------------------------| | yield 表達式 yield表達式結果 # 執行順序流程相似 $res = (yield 1); // <- var_dump("1::" . $gen->current()); 第一步到yield返回 var_dump($res); // <- $gen->send(222); 第二步send:222後,繼續往下走$res=222 而後var_dump($res), 而後到了yield 2 $res = (yield 2); // <- var_dump("2::" . $gen->current()); 打印當前的值2 var_dump($res); // <- $gen->send(333); 第三步send:333後,繼續往下走$res=333 而後var_dump($res), 而後到了yield 3 $res = (yield 3); // <- var_dump("3::" . $gen->current()); var_dump($res); // <- $gen->send(null); 第二步send:null後,繼續往下走$res=null 而後var_dump($res), 而後到了yield 4 $res = (yield 4); // <- var_dump("4::" . $gen->current());
# 內部定義異常並返回,外部接收 function gen() { echo "Gen 開始\n"; yield new Exception('內部定義異常'); echo "Gen 結束\n"; } $gen = gen(); try { throw $gen->current(); } catch (\Exception $e) { echo "外部捕獲異常:" . $e->getMessage() . PHP_EOL; } $gen->send("123"); # output: Gen 開始 外部捕獲異常:內部定義異常 Gen 結束 # 內部接收send傳入的異常,而後直接throw,外部接收 function gen() { echo "Gen 開始\n"; throw (yield new Exception('內部定義異常')); echo "Gen 結束\n"; } $gen = gen(); try { throw $gen->current(); } catch (\Exception $e) { echo "外部捕獲異常:" . $e->getMessage() . PHP_EOL; } try { $gen->send(new \Exception("外部定義異常")); } catch (\Exception $e) { echo "外部捕獲異常:" . $e->getMessage() . PHP_EOL; } # output Gen 開始 外部捕獲異常:內部定義異常 外部捕獲異常:外部定義異常 # function gen() { echo "Gen 開始\n"; try { yield new Exception('內部定義異常'); } catch (Exception $e) { echo "內部捕獲異常:" . "Exception: {$e->getMessage()}\n"; } echo "Gen 結束\n"; } $gen = gen(); try { throw $gen->current(); } catch (\Exception $e) { echo "外部捕獲異常:" . $e->getMessage() . PHP_EOL; } // 注意這裏是throw方法,至關於 // $gen->send(new \Exception("外部定義異常")); // throw yield (yield接收 = new \Exception("外部定義異常")) $gen->throw(new \Exception("外部定義異常")); #output Gen 開始 外部捕獲異常:內部定義異常 內部捕獲異常:Exception: 外部定義異常 Gen 結束
初識咱們只須要先理解next和send便可 next->讓咱們能夠主動自動執行迭代器 send->能夠讓咱們的迭代器實現雙向通訊,改變執行體流程順序 後續咱們會介紹使用場景和Co自動執行體等
2018.09.20: [PHP下的生成器嘗試一:初識PHP下的生成器] -> [PHP下的異步嘗試一:初識生成器]