天生器是 PHP 5.5 引入的新特性,但是目测很少人用到它,实在这是个非常有用的功能。

天生器和迭代器有点类似,但是与标准的PHP迭代器不同,PHP天生器不哀求类实现Iterator接口,从而减轻了类的开销和包袱。
天生器会根据需求每次打算并产出须要迭代的值,这对运用的性能有很大的影响:试想如果标准的PHP迭代器常常在内存中实行迭代操作,这要预先打算出数据集,性能低下;如果要利用特定办法打算大量数据,如操作Excel表数据,对性能影响愈甚。
此时我们可以利用天生器,即时打算并产出后续值,不占用宝贵的内存空间。

2、创建天生器

phpfeofphp生成器的创立和应用 RESTful API

天生器的创建办法很大略,由于天生器便是PHP函数,只不过要在函数中一次或多次利用yield关键字。
与普通的PHP函数不同的是,天生器从不返回值,只产出值。
下面是一个大略的天生器实现:

function getLaravelAcademy() { yield 'http://xxx.xxx'; yield 'xxx'; yield 'xxx';}

很大略吧!
调用此生成器函数时,PHP会返回一个属于Generator类的工具,这个工具可以利用foreach函数迭代,每次迭代,PHP会哀求Generator实例打算并供应下一个要迭代的值。
天生器的优雅表示在每次产出一个值之后,天生器的内部状态都会停顿;向天生器要求下一个值时,内部状态又会规复。
天生器内部的状态会一贯在停顿和规复之间切换,直到抵达函数定义体的末端或碰着空的return语句为止。
我们可以利用下面的代码调用并迭代上面定义的天生器:

foreach(getLaravelAcademy() as $yieldedValue) {

echo $yieldedValue, PHP_EOL;

}

3、利用天生器

下面我们实现一个大略的函数用于天生一个范围内的数值,以此解释天生器是如何节省内存的。
首先我们通过迭代器来实现:

function makeRange($length) { $dataSet = []; for ($i=0; $i<$length; $i++) { $dataSet[] = $i; } return $dataSet;}$customRange = makeRange(1000000);foreach ($customRange as $i) { echo $i . PHP_EOL;}

此时实行会报错,提示超出单个PHP进程内存限定(要为100万个数字供应内存空间):

下面我们来改进实现方案,利用天生器实现如下:

function makeRange($length) { for ($i=0; $i<$length; $i++) { yield $i; }}foreach (makeRange(1000000) as $i) { echo $i . PHP_EOL;}

再次实行就可以毫无压力的打印出结果,由于天生器每次只须要为一个整数分配内存。

此外,一个常用的利用案例便是利用天生器迭代流资源(文件、音频等)。
假设我们想要迭代一个大小为4GB的CSV文件,而虚拟私有做事器(VPS)只许可PHP利用1GB内存,因此不能把全体文件都加载到内存中,下面的代码展示了如何利用天生器完成这种操作:

function getRows($file) { $handle = fopen($file, 'rb'); if ($handle == FALSE) { throw new Exception(); } while (feof($handle) === FALSE) { yield fgetcsv($handle); } fclose($handle);}foreach ($getRows($file) as $row) { print_r($row);}

上述示例一次只会为CSV文件中的一行分配内存,而不会把全体4GB的CSV文件都读取到内存中。

4、总结

天生器是功能多样性和简洁性之间的折上钩划,天生器只是向提高的迭代器,这意味着不能利用天生器在数据集中实行退却撤退、快进或查找操作,只能让天生器打算并产出下一个值。
迭代大型数据集或数列时最适宜利用天生器,由于这样占用的系统内存最少。
天生器也能完成迭代器能完成的大略任务,而且利用的代码更少。