PHP生成器-动态生成内容的数组

定义:

1.生成器:“动态生成内容的数组”,用到值的时候才去生产。

2.搭配使用:关键字 yield foreach

过程:

1.作为生成器的方法相当于定义了一个数组;

2.生成器中的 yield,每次出现相当于定义了一个数组中出现的值;

3.foreach 会遍历生成器中所有的 yield 定义的值(某个被定义的值(表达式),只有在被用到的时候,当前yield同上一个yield之间的其他表达式才会被执行,当前yield之后的表达式不会被执行);

4.foreach 正常结束的话(不是break),判断一下生成器中最后一个yield之后是否有其他表达式未执行完,是的话,执行之;

5.不同于线程,不是并行执行,只是在主业务逻辑和生成器之间来回切换执行,只节省了空间,时间没省下来。

PHP生成器-动态生成内容的数组

设想使用场景:

1、取大数据文件,生成器中逐行读取;

2、???

例子:

gen.php

  1. function gen()
  2. {
  3. echo '生成器开始执行' . PHP_EOL;
  4. for ($i = 0; $i < 5; $i++) {
  5. echo '产生数据之前:' . $i . PHP_EOL;
  6. yield $i;
  7. echo '产生数据之后:' . $i . PHP_EOL;
  8. }
  9. echo '再来一个数据' . PHP_EOL;
  10. yield 5;
  11. echo '生成器执行结束' . PHP_EOL;
  12. }
  13. $gen_func = gen();
  14. echo '生成器开始执行了吗?' . PHP_EOL;
  15. foreach ($gen_func as $key => $val) {
  16. echo '使用数据前' . PHP_EOL;
  17. echo '使用数据:' . $val . PHP_EOL;
  18. echo '使用数据后' . PHP_EOL;
  19. //if ($key >= 4) {
  20. //break;
  21. //}
  22. }

执行结果:

php gen.php

  1. 生成器开始执行了吗?
  2. 生成器开始执行
  3. 产生数据之前:0
  4. 使用数据前
  5. 使用数据:0
  6. 使用数据后
  7. 产生数据之后:0
  8. 产生数据之前:1
  9. 使用数据前
  10. 使用数据:1
  11. 使用数据后
  12. 产生数据之后:1
  13. 产生数据之前:2
  14. 使用数据前
  15. 使用数据:2
  16. 使用数据后
  17. 产生数据之后:2
  18. 产生数据之前:3
  19. 使用数据前
  20. 使用数据:3
  21. 使用数据后
  22. 产生数据之后:3
  23. 产生数据之前:4
  24. 使用数据前
  25. 使用数据:4
  26. 使用数据后
  27. 产生数据之后:4
  28. 再来一个数据
  29. 使用数据前
  30. 使用数据:5
  31. 使用数据后
  32. 生成器执行结束

部分执行过程解读:

1、14行并没有调用生成器gen(),只是做了定义;

2、进入 foreach 循环后,开始调用gen();

3、$val 需要值的时候,记住当前位置a,执行生成器;

4、执行到 yield 定义的地方,取到值了,记住当前位置b,返回位置a;

5、循环3、4两步;

6、循环结束,执行完 yield 最后一次定义的地方的之后的剩下部分;

7、去掉20、21、22行的注释后执行,"使用数据:4 使用数据后",就不再有其他输出。

其他(仅验证了PHP7.1.14版本)

1、value);//YES value;//NO

2、可以使用yield val;

3、PHP7之后,yield from 可以调用生成器、数组、使用return等。