定义:
生成器:“动态生成内容的数组”,用到值的时候才去生产。
搭配使用:关键字 yield foreach
过程:
- 作为生成器的方法相当于定义了一个数组;
- 生成器中的 yield,每次出现相当于定义了一个数组中出现的值;
- foreach 会遍历生成器中所有的 yield 定义的值(某个被定义的值(表达式),只有在被用到的时候,当前yield同上一个yield之间的其他表达式才会被执行,当前yield之后的表达式不会被执行);
- foreach 正常结束的话(不是break),判断一下生成器中最后一个yield之后是否有其他表达式未执行完,是的话,执行之;
- 不同于线程,不是并行执行,只是在主业务逻辑和生成器之间来回切换执行,只节省了空间,时间没省下来。
设想使用场景:
1、取大数据文件,生成器中逐行读取;
2、???
例子:
gen.php
1 <?php
2 function gen()
3 {
4 echo '生成器开始执行' . PHP_EOL;
5 for ($i = 0; $i < 5; $i++) {
6 echo '产生数据之前:' . $i . PHP_EOL;
7 yield $i;
8 echo '产生数据之后:' . $i . PHP_EOL;
9 }
10 echo '再来一个数据' . PHP_EOL;
11 yield 5;
12 echo '生成器执行结束' . PHP_EOL;
13 }
14 $gen_func = gen();
15 echo '生成器开始执行了吗?' . PHP_EOL;
16 foreach ($gen_func as $key => $val) {
17 echo '使用数据前' . PHP_EOL;
18 echo '使用数据:' . $val . PHP_EOL;
19 echo '使用数据后' . PHP_EOL;
20 //if ($key >= 4) {
21 //break;
22 //}
23 }
执行结果:
php gen.php
生成器开始执行了吗?
生成器开始执行
产生数据之前:0
使用数据前
使用数据:0
使用数据后
产生数据之后:0
产生数据之前:1
使用数据前
使用数据:1
使用数据后
产生数据之后:1
产生数据之前:2
使用数据前
使用数据:2
使用数据后
产生数据之后:2
产生数据之前:3
使用数据前
使用数据:3
使用数据后
产生数据之后:3
产生数据之前:4
使用数据前
使用数据:4
使用数据后
产生数据之后:4
再来一个数据
使用数据前
使用数据:5
使用数据后
生成器执行结束
部分执行过程解读:
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、$data = (yield $value);//YES $data = yield $value;//NO
2、可以使用yield $key => $val;
3、PHP7之后,yield from 可以调用生成器、数组、使用return等。
(传送门:PHP手册里都有)
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。