1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 |
public function debugYield() { $fibs = $this->fib(3); /** foreach 形式调用; * 解析 foreach $fibs 的执行逻辑 * 第一次: ->rewind, ->valid, ->current, ->key, 执行循环体逻辑 * 第二次:->next, ->valid, ->current, ->key, 执行循环体逻辑 * 第三次:->next, ->valid, ->current, ->key, 执行循环体逻辑 * 最后一次:->next, ->valid, * 注:从上面可以发现,foreach的调用次数是数组长度 + 1,从而判断数组已经没有更多元素结束循环 foreach ($fibs as $fib) { echo "值:$fib, 执行顺序 2\n"; } */ // while 循环实现上面的 foreach 逻辑 // $fibs->rewind(); // 该语句可省略,因为生成迭代对象的时候已经隐含地执行了rewind操作 while(true) { if($fibs->valid()) { $value = $fibs->current(); echo "值:$value, 执行顺序 2\n"; } else { break; } $fibs->next(); } } /** * 函数说明:含有 yield 关键字的函数为生成器函数,生成器函数为特殊函数 * 只能 return 空,他表示终止生成器继续执行。 * @param $n * @return \Generator */ public function fib($n) { $cur = 1; $prev = 0; for ($i = 0; $i < $n; $i++) { echo "执行顺序 1\n"; /** * 生成器函数的核心是yield关键字。它最简单的调用形式看起来像一个return申明, * 不同之处在于普通return会返回值并终止函数的执行,而yield会返回一个值给循环调用此生成器的代码[code] * 并且只是暂停执行生成器函数。(这里把控制权交给了[code]) * 虽然 yield 会将控件权交给了[code],但 yield 所在的函数状态会被保存起来, * 当 [code] 执行 next 方法时会还原函数状态,并执行 yield 之后的逻辑 */ yield $cur; echo "执行顺序 3\n"; $temp = $cur; $cur = $prev + $cur; $prev = $temp; } } |