生成器提供了一种更容易的方法来实现简单的对象迭代,生成器允许你在 foreach 代码块中写代码来迭代一组数据而不需要在内存中创建一个数组, 那会使你的内存达到上限,或者会占据可观的处理时间。相反,你可以写一个生成器函数,就像一个普通的自定义函数一样, 和普通函数只返回一次不同的是, 生成器可以根据需要 yield 多次,以便生成需要迭代的值。
1:生成器会对PHP应用的性能有非常大的影响
2:PHP代码运行时节省大量的内存
3:比较适合计算大量的数据
生成器函数的核心是yield关键字。它最简单的调用形式看起来像一个return申明,不同之处在于普通return会返回值并终止函数的执行,而yield会返回一个值给循环调用此生成器的代码并且只是暂停执行生成器函数。
我们都知道php中有一个range函数,他的作用是创建一个包含指定范围的元素的数组,他的基本语法为range(low,high,step),表示包含从 low 到 high (包含 low 和 high)之间的整数或字符。如果 high 比 low 小,则返回反序的数组,step表示步长默认步长为1
例如:
range(0,10)
输出结果为:
Array ( [0] => 0 [1] => 1 [2] => 2 [3] => 3 [4] => 4 [5] => 5 [6] => 6 [7] => 7 [8] => 8 [9] => 9 [10] => 10 )
但是如果我们想要获取1-10000000之间所有的数组的话,调用range(0,10000000)时将导致占用内存耗尽
这时候我们就可以使用php的生成器来实现
public function rangeText($start, $limit, $step = 1)
{
if ($start < $limit) {
if ($step <= 0) {
throw new InvalidConfigException('步长至少大于0');
}
for ($i = $start; $i <= $limit; $i += $step) {
yield $i;
}
} else {
if ($step >= 0) {
throw new InvalidConfigException('步长至少大于0');
}
for ($i = $start; $i >= $limit; $i += $step) {
yield $i;
}
}
}
foreach ($this->rangeText(0,10000000) as $number) {
echo $number .'<br />';
}
调用生成函数之后我们就可以将0-10000000之间的所有数打印出来
PHP开发很多时候都要读取大文件,比如csv文件、text文件,或者一些日志文件。这些文件如果很大,比如10个G。这时,直接一次性把所有的内容读取到内存中计算不太现实。那么这里生成器就可以派上用场
例:使用生成器读取txt文件
public function readTest($txt)
{
//打开文件
$handle = fopen($txt, 'rb');
//检测是否已到达文件末尾
while (feof($handle)===false) {
//使用生成器逐行读取文件数据
yield fgets($handle);
}
//关闭打开的文件
fclose($handle);
}
foreach ($this->readTest('test.txt') as $key=>$value) {
echo $value.'<br />';
}
这时候我们就可以在浏览器看到txt文件中内容了