欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  php教程

php生成器的使用 - coder5

程序员文章站 2022-04-15 08:40:33
...
按照php的文档说明

一个生成器函数看起来像一个普通的函数,不同的是普通函数返回一个值,而一个生成器可以yield生成许多它所需要的值。
当一个生成器被调用的时候,它返回一个可以被遍历的对象.当你遍历这个对象的时候(例如通过一个foreach循环),PHP 将会在每次需要值的时候调用生成器函数,并在产生一个值之后保存生成器的状态,这样它就可以在需要产生下一个值的时候恢复调用状态。
一旦不再需要产生更多的值,生成器函数可以简单退出,而调用生成器的代码还可以继续执行,就像一个数组已经被遍历完了。

一个生成器不可以返回值: 这样做会产生一个编译错误。然而return空是一个有效的语法并且它将会终止生成器继续执行。

/**
 * 安原来的写法,我需要一个方法来整理数据。nums()会返回一个数组或者其他可以迭代的数据
 * 然后再遍历这个数组
 */
function nums() {
    $array = array();
    for ($i = 0; $i $i) {
        $array[]= $i;
    }
    return $array;
}

foreach (nums() as $v){
    var_dump($v);
};


/**
 * 但是用了yield之后,我不再需要创建一个变量,来存储这个数据。
 * 而是在内部会为生成的值配对连续的整型索引,就像一个非关联的数组。 
 * 这样会省掉很大的内存开销
 */
function nums2() {
    for ($i = 0; $i $i) {
        yield $i;
    }
}

foreach (nums2() as $v){
    var_dump($v);
};

如果需要在生成器中指定下标 可以 yield $id => $fields;

$handler=function() {
        $start = microtime(true);

        yield;

        $use_time = (microtime(true) - $start) * 1000;
        echo "the time is ".(int)$use_time."ms\n";
        
        yield;
        $use_time = (microtime(true) - $start) * 1000;
        echo "the time2 is ".(int)$use_time."ms\n";
};

//根据文档的说明 当一个生成器被调用的时候,它返回一个可以被遍历的对象.当你遍历这个对象的时候
//PHP 将会在每次需要值的时候调用生成器函数,
//也就是说此时并没有真的调用$handler中定义的函数。而是返回了一个生成器
$generator = call_user_func_array($handler,array());

if ($generator && $generator instanceof \Generator) {

    //当对生成器进行迭代的时候,才会真正的调用该$handler中定义的函数
    if ($generator->current() === false) {
        /***do some thing**/
    }
}

生成器也可以通过引用来使用

/**
 * 使用引用来生成值
 */
function &gen_reference() {
    $value = 3;

    while ($value > 0) {
        yield $value;
    }
}
/* 
 * 我们可以在循环中修改$number的值,而生成器是使用的引用值来生成,所以gen_reference()内部的$value值也会跟着变化。
 */
foreach (gen_reference() as &$number) {
    echo (--$number).'... ';
}

除了通过引用来改变生成器中的数据之外,我们还可以使用send方法传递数据

function printer() {
    while (true) {
        $string = yield;
        echo $string;
    }
}

$printer = printer();
$printer->send('Hello world!');

但是上面的例子如果没有 while(true),那么无论后面send多少次,该生成器只会执行一次。如此说来,似乎可以用这个来控制本函数调用的最多次数?

除了向里面传递数据之外,还可以throw异常。

function application() {
    while (true){
        try {
             yield;

        }catch (Exception $e){
            echo $e;
        }
    }
}

$printer = application();
$printer->throw(new  Exception("test"));

生成器的嵌套

/**
 * 在php7中 可以通过yield from 进行嵌套
 */
function count_to_ten() {
    yield 1;
    yield 2;
    yield from [3, 4];
    yield from new ArrayIterator([5, 6]);
    yield from seven_eight();
    yield 9;
    yield 10;
}

function seven_eight() {
    yield 7;
    yield from eight();
}

function eight() {
    yield 8;
}

$gen = count_to_ten();
foreach ($gen as $num) {
    echo "$num ";
}
echo "the return value is ".$gen->getReturn();