php中Generator的执行过程

说到php中的Generator(生成器),有人可能会想到协程,这里我们先不说php如何实现协程,我们探究下Generator的执行过程。
Generator是通过yield实现,yield 关键字是php5.5版本推出的一个特性。
首先,看下面的代码:

                function gen(){
                  while(true){
                    yield "gen\n";
                  }
                }
                
                $gen = gen();
                echo "Generator"; 
                

如果没有了解过yield的话,你会认为上面代码执行的结果是:死循环。但实际上,它会echo出Generator。

到这里,也许你会觉得奇怪,yield怎么可以结束循环?下面就为大家说明一下:

Generator提供的方法:

Generator::current — 返回当前产生的值
Generator::key — 返回当前产生的键
Generator::next — 生成器继续执行
Generator::rewind — 重置迭代器
Generator::send — 向生成器中传入一个值
Generator::throw — 向生成器中抛入一个异常
Generator::valid — 检查迭代器是否被关闭
Generator::__wakeup — 序列化回调

生成器提供了一种更容易的方法来实现简单的对象迭代(迭代器),相比较定义类实现 Iterator 接口的方式,性能开销和复杂性大大降低。

列子:

              function gen(){
                   for($i=0;$i<5;$i++)
                   {
                       echo (yield $i).$i.'
'; } } $gen = gen(); foreach($gen as $k=>$v){ echo "{$k}---{$v}".'
'; }

结果是:

php中Generator的执行过程_第1张图片

从上面的结果,我们可以分析出以下几点:
1当Generator对象被foreach的时候,内部的valid,current,key方法会依次被调用,其返回值是foreach语句的value和key。
2循环的终止条件则根据valid方法的返回而定。如果返回的是true则继续循环,如果是false则终止整个循环,结束遍历。
3一次循环体结束之后,将调用next进行下一次的循环直到valid返回false。而rewind方法则是在整个循环开始前被调用(也就是生成Generator对象时),这样保证了我们多次遍历得到的结果都是一致的。

下面我们来证明一下这个流程:

$gen = gen();
echo $gen->key();//结果是0,生成Generator对象时,rewind已经执行。
echo $gen->key().'----'.$gen->current();// 0----0
var_dump($gen->next());//var_dump值是null,但是还会echo出多一个0;这个0是怎样来的呢?原因是:next()执行后,第1个yield到第二个yieldz之间的的语法被执行,即是:echo (yield $i).$i.'
';由于next()是没有返回值,即(yield $i)这个表达式没有值,而$i的值是0;
echo $gen->key().'----'.$gen->current();// 1----1 目前是第2个yield

上面这个例子可以证明,Generator内部的流程,特别注意next()的理解。

最后,我们说一下,send():

官方解析:向生成器中传入一个值,并且当做 yield 表达式的结果,然后继续执行生成器。如果当这个方法被调用时,生成器不在 yield 表达式,那么在传入值之前,它会先运行到第一个 yield 表达式。

翻译下的结论是:
send()方法主要用于发送数据给当前yield,即yield表达式被当作一个值被替换,且继续执行下一个yield,即next()

证明例子:
$gen = gen();
$gen->send(666);//6660

6660结果分析:首先把666代替当前yield表达式的值,然后执行next(),即运行echo (yield $i).$i.'
',当前yield是666,所以最终结果是:6660。注意与next()的区别!!!

总结:
1.yield只能用于函数内部,在非函数内部运用会抛出错误。
2.如果函数包含了yield关键字的,那么函数执行后的返回值永远都是一个Generator对象。
3.如果函数内部同事包含yield和return 该函数的返回值依然是Generator对象,但是在生成Generator对象时,return语句后的代码被忽略。
4.Generator类实现了Iterator接口。
5.可以通过返回的Generator对象内部的方法,获取到函数内部yield后面表达式的值。
6.可以通过Generator的send方法给yield 关键字赋一个值。
7.一旦返回的Generator对象被遍历完成,便不能调用他的rewind方法来重置。
8.Generator对象不能被clone关键字克隆 。

实际应用:
1.协程
2.Genenrator返回的是迭代器,在处理大数据的时候不用一次性的加载到内存中,可看http://php.net/manual/zh/lang...

你可能感兴趣的:(php,yield)