在讲解生成器以前先介绍一下迭代器:php
在 PHP 中,一般状况下遍历数组使用 foreach 来遍历。html
若是咱们要想让一个对象能够遍历呢?laravel
PHP 为咱们提供了 Iterator 接口,只要实现了这个接口,这个对象就能够经过 foreach 来迭代。面试
例子以下:数组
class myIterator implements Iterator { private $index = 0; private $data = ''; public function __construct($data) { $this->index = 0; $this->data = $data; } function rewind() { $this->index = 0; } function current() { return $this->data[$this->index]; } function key() { return $this->index; } function next() { ++$this->index; } function valid() { return isset($this->data[$this->index]); } } $it = new myIterator(array( "hello", "php", "iterator", )); foreach($it as $key => $value) { echo "$key : $value<br>"; }
咱们经过foreach遍历 $it 时,PHP 会本身依次调用:函数
rewind() 重置到第一个元素
valid() 检查当前位置是否有效
current() 返回当前元素
key() 返回当前元素的键
next() 指向下一个元素post
生成器是 PHP 5.5 引入的新特性,可是目前貌似不多人用到它。
下面试 PHP 官方文档上对生成器的解释:
生成器提供了一种更容易的方法来实现简单的对象迭代,相比较定义类实现 Iterator 接口的方式,性能开销和复杂性大大下降。
生成器容许你在 foreach 代码块中写代码来迭代一组数据而不须要在内存中建立一个数组, 那会使你的内存达到上限,或者会占据可观的处理时间。相反,你能够写一个生成器函数,就像一个普通的自定义函数同样, 和普通函数只返回一次不一样的是, 生成器能够根据须要 yield 屡次,以便生成须要迭代的值。性能
为了体现生成器的有点,下面咱们定义一个函数来进行比较:this
function func1() { foreach (range(0, 1000000) as $value){ echo $value; } } func1(); // ( ! ) Fatal error: Allowed memory size of 134217728 bytes exhausted (tried to allocate 402653192 bytes) in xxx.php on line 5
由于建立如此大的数组到内存中进行迭代,则 PHP 直接提示超出了单个进程的内存限制。.net
下面咱们换作生成器的方式来处理:
function func1() { foreach (range(0, 1000000) as $value){ yield $value; } } var_dump(func1()); // object(Generator)[1] foreach (func1() as $value){ echo $value; }
能够看到咱们调用 func1() 返回了一个 Generator 对象,这个对象可使用 foreach 迭代,每次迭代,PHP 会要求 Generator 实例计算并提供下一个要迭代的值。生成器的优雅体如今每次产出一个值以后,生成器的内部状态都会停顿;向生成器请求下一个值时,内部状态又会恢复。生成器内部的状态会一直在停顿和恢复之间切换,直到抵达函数定义体的末尾或遇到空的return语句为止。
参考连接:
https://laravelacademy.org/po...
http://php.net/manual/zh/lang...