ホームページ  >  記事  >  バックエンド開発  >  PHPのジェネレーターとは何ですか

PHPのジェネレーターとは何ですか

巴扎黑
巴扎黑オリジナル
2017-08-16 09:48:504767ブラウズ

PHP のジェネレーターは、PHP5.5 以降に追加された新機能で、中国語では「ジェネレーター」と訳されます。ジェネレーターを使用してイテレーターを実装する場合、イテレーター インターフェースを継承するクラスを作成する必要はありません。

PHPのジェネレーターとは何ですか

ジェネレーターとは

PHP 5.5 以降、PHP には Generator という新しい機能が追加されました。中国語では Generator code> と翻訳されます。 。 <code>Generator,中文译为生成器

Generator提供了一种方便的实现简单的Iterator(迭代器)的方式,使用Generator实现Iterator不需要创建一个类来继承Iterator接口。

生成器可以简单地用来实现对象的迭代,让我们先从官方的一个小例子说起。

xrange

在 PHP 中,我们都知道,有一个函数叫做 range,用来生成一个等差数列的数组,然后我们可以用这个数组进行 foreach 的迭代。具体就想这样。

foreach (range(1, 100, 2) as $num) {    echo "{$num}\n";
}

这一段代码就会输出首项为 1,末项为 100,公差为 2 的等差数列。它的执行顺序是这样的。首先,range(1, 100, 2) 会生成一个数组,里面存了上面那样的一个等差数列,之后在 foreach 中对这个数组进行迭代。

那么,这样就会出现一个问题,如果我要生成 100 万个数字呢?那我们就要占用上百兆内存。虽然现在内存很便宜,但是我们也不能这么浪费内存嘛。那么这时,我们的生成器就可以排上用场了。考虑下面的代码。

function xrange($start, $limit, $step = 1) {    yield $start;
    $start++;
}foreach (xrange(1, 100, 2) as $num) {    echo "{$num}\n";
}

这段代码所的出来的结果,和前面的那段代码一模一样,但是,它内部的原理是天翻地覆了。

我们刚才说了,前面的代码,range 会生成一个数组,然后 foreach 来迭代这个数组,从而取出某一个值。但是这段代码呢,我们重新定义了一个 xrange 函数,在函数中,我们用了一个关键字 yield。我们都知道定义一个函数,希望它返回一个值得时候,用 return 来返回。那么这个 yield 呢,也可以返回一个值,但是,它和 return 是截然不同的。

使用 yield 关键字,可以让函数在运行的时候,中断,同时会保存整个函数的上下文,返回一个 Generator 类型的对象。在执行对象的 next 方法时,会重新加载中断时的上下文,继续运行,直到出现下一个 yield 为止,如果后面没有再出现 yield,那么就认为整个生成器结束了。

这样,我们上面的函数调用可以等价地写成这样。

$nums = xrange(1, 100, 2);while ($nums->valid()) {    echo $nums->current() . "\n";
    $nums->next();
}

在这里,$num 是一个 Generator 的对象。我们在这里看到三个方法,validcurrentnext。当我们函数执行完了,后面没有 yield 中断了,那么我们在 xrange 函数就执行完了,那么 valid 方法就会变成 false。而 current 呢,会返回当前 yield 后面的值,这是,生成器的函数会中断。那么在调用 next 方法之后,函数会继续执行,直到下一个 yield 出现,或者函数结束。

好了,到这里,我们看到了通过 yield 来“生成”一个值并返回。其实,yield 其实也可以这么写 $ret = yield;。同返回值一样,这里是将一个值在继续执行函数的时候,传值进函数,可以通过 Generator::send($value) 来使用。例如。

function sum(){
    $ret = yield;    echo "{$ret}\n";
}

$sum = sum();
$sum->send('I am from outside.');

这样,程序就会打印出 send 方法传进去的字符串了。在 yield

Generator は、単純な Iterator を実装するための便利な方法を提供します。Generator を使用して Iterator を実装する場合、Iterator インターフェイスを継承するクラスを作成する必要はありません。 🎜🎜xrange🎜🎜🎜 PHP では、算術シーケンスの配列を生成するために使用される range と呼ばれる関数があることを誰もが知っており、この配列を使用して foreach の繰り返し。具体的にはこれがやりたいことです。 🎜🎜🎜🎜

🎜🎜🎜🎜

function xrange($start, $limit, $step = 1) {
    $ret = yield $start;
    $start++;    echo "{$ret}\n";
}
🎜 このコードは、最初の項が 1、最後の項が 100、許容差が 2 の等差数列を出力します。実行シーケンスは以下の通りです。まず、range(1, 100, 2) は、上記のように算術シーケンスを格納する配列を生成し、この配列を foreach で繰り返します。 🎜🎜そこで、100 万個の数値を生成したい場合はどうすればよいかという疑問が生じます。そうなると、数百メガバイトのメモリを占有することになります。今はメモリが非常に安くなりましたが、このようにメモリを無駄にするわけにはいきません。そんなとき、私たちの発電機が役に立ちます。次のコードを考えてみましょう。 🎜🎜🎜🎜

🎜🎜🎜🎜rrreee🎜 このコードの結果は前のコードとまったく同じですが、内部原理は完全に異なります。 🎜🎜前のコードでは、range が配列を生成し、次に foreach が配列を反復して特定の値を取り出すと述べました。ただし、このコードでは、xrange 関数を再定義し、その関数内でキーワード yield を使用しました。関数を定義し、それが値を返すことを期待する場合、値を返すには return を使用することは誰もが知っています。したがって、この yield も値を返すことができますが、それは return とは完全に異なります。 🎜🎜 yield キーワードを使用すると、実行中の関数を中断し、関数全体のコンテキストを保存し、Generator 型のオブジェクトを返すことができます。オブジェクトの next メソッドが実行されると、中断時のコンテキストが再ロードされ、 がない場合は次の <code>yield が表示されるまで操作が継続されます。 yield が表示されると、ジェネレーター全体が終了したとみなされます。 🎜🎜このようにして、上記の関数呼び出しは等価的に次のように書くことができます。 🎜🎜🎜🎜

🎜🎜🎜🎜rrreee🎜 ここで、$numGenerator のオブジェクトです。ここでは、validcurrent、および next という 3 つのメソッドが表示されます。関数が実行され、後で yield 割り込みが発生しない場合、xrange 内の関数が実行され、valid メソッドは 偽コード>。 current に関しては、現在の yield の後の値が返されます。これは、ジェネレーター関数が中断されることを意味します。 next メソッドを呼び出した後、次の yield が表示されるか関数が終了するまで、関数は実行を続けます。 🎜🎜わかりました。ここでは、yield が値を「生成」し、それを返すために使用されていることがわかります。実際、yield$ret = yield; と書くこともできます。戻り値と同様に、関数の実行を継続するときに値が関数に渡されます。これは Generator::send($value) を通じて使用できます。例えば。 🎜🎜🎜🎜

🎜🎜🎜🎜rrreee🎜 このようにして、プログラムは send メソッドによって渡された文字列を出力します。 yield の両側で同時に呼び出しを行うことができます。 🎜🎜🎜🎜

function xrange($start, $limit, $step = 1) {
    $ret = yield $start;
    $start++;    echo "{$ret}\n";
}

而像这样的使用,send() 可以返回下一个 yield 的返回。

其它的 Generator 方法

Generator::key()

对于 yield,我们可以这样使用 yield $id => $value,这是,我们可以通过 key 方法来获取 $id,而 current 方法返回的是 $value

Generator::rewind()

这个方法,可以帮我们让生成器重新开始执行并保存上下文,同时呢,会返回第一个 yield 返回的内容。在第一次执行 send 方法的时候,rewind 会被隐式调用。

Generator::throw()

这个方法,向生成器中,抛送一个异常。

后记

yield 作为 PHP 5.5 的新特性,让我们用了新的方法来高效地迭代数据。同时,我们还可以使用 yield 来实现协程。

更多相关知识,请访问 PHP中文网!!

以上がPHPのジェネレーターとは何ですかの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。