foreach 的执行逻辑解析和关键字yield的理解

  • Post author:
  • Post published:10月 22, 2019
  • Post category:PHP
  • Post comments:0评论

    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;
        }
    }

发表评论