PHP中的 Iterator 与 Generator

在讲解生成器以前先介绍一下迭代器: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...

相关文章
相关标签/搜索