官方解释yield
yield生成器
是PHP5.5
之后出现的,官方文档这样解释:yield
提供了一种更容易的方法来实现简单的迭代对象,相比较定义类实现 Iterator
接口的方式,性能开销和复杂性大大降低。
生成器的核心是一个yield
关键字,一个生成器函数看起来像一个普通的函数,不同的是:普通函数返回一个值,而一个生成器可以yield
生成许多它所需要的值。生成器函数被调用时,返回的是一个可以被遍历的对象。
yield
和return
有点类似,不过不同的是,return
会返回值并且终止代码的执行,而yield
会返回一个值给循环调用此生成器的代码并且只是暂停执行生成器函数。
意思即是 把数据一行行 读取到PHP运行内存,并非一次性读取到PHP运行内存,众所周知,PHP有很多内置函数,可以帮助我们对数据进行加工操作,因为数据都在内存里面,所以能操作,但是PHP的运行内存是有极限,默认128M。
注意:因为非缓冲查询是 会长时间连接数据库的,有可能会造成慢查询、锁表之类的情况,比较耗MysqL资源
如果用缓存查询,PHP内存就会直接爆了,出现内存不足的情况。好了,这里主要是为了突出yield
yield性能
比较适合计算大量的数据
yield运用
生成器允许你在 foreach 代码块中写代码来迭代一组数据而不需要在内存中创建一个数组,那会使你的内存达到上限,或者会占据可观的处理时间。相反,你可以写一个生成器函数,就像一个普通的自定义函数一样, 和普通函数只返回一次不同的是, 生成器可以根据需要 yield 多次,以便生成需要迭代的值。
例子讲解
我这里只是构建了一个数组来给大家演示,平常你在操作数据库输出数据也是一样的。转为array
//仓库库存扣除测试 public function cangku_stock() { //set_time_limit(0); //表示永久运行,这里我是测试array的时候用到的 $order_info = $this->read_temp_api_order_info(10); //这里我就测试了10条数据,效果是看不出来的 foreach($order_info as $temp_api_order_info){ dd($temp_api_order_info); //打印出来看看数据 //处理数据 $api_ware_id = $this->o->getCangkuApiUrl() .'ware/program/addOutWare'; $out_wares = api_request($api_ware_id, $temp_api_order_info); $temp_out_wares = json_decode($out_wares, true); if ($temp_out_wares['code'] != 1) { $msg = (isset($temp_out_wares['msg']) && $temp_out_wares['msg']) ? $temp_out_wares['msg'] : var_export($out_wares, true); throw new Exception($msg); } } //dd(批量更新成功.date('Y-m-d H:i:s'));
可以看到我们调用 $order_info = $this->read_temp_api_order_info(10);返回了一个 Generator 对象,这个对象可以使用 foreach 迭代,每次迭代,PHP 会要求 Generator 实例计算并提供下一个要迭代的值。
生成器的优雅体现在每次产出一个值之后,生成器的内部状态都会停顿;向生成器请求下一个值时,内部状态又会恢复。生成器内部的状态会一直在停顿和恢复之间切换,直到抵达函数定义体的末尾或遇到空的 return 语句为止。
效果如下:
这里测试大量数据,直接更改$this->read_temp_api_order_info(10);就
好,如果是计算数据表数量,那你就要改改这个方法了。自己试着更改一下吧。
这里我主要给大家讲解yield的用法,如果要看插入数据表用了多长时间,自己可以在数据表增加插入时间的字段,再看看第一条数据插入与最后一条数据插入的时候的对比的。