# 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下的异步尝试一:初识生成器]