PHP 8.3.27 Released!

Generator::send

(PHP 5 >= 5.5.0, PHP 7, PHP 8)

Generator::sendSendet einen Wert an einen Generator

Beschreibung

public Generator::send(mixed $value): mixed

Sendet den angegebenen Wert als Ergebnis des aktuellen yield-Ausdrucks an den Generator und setzt die Ausführung des Generators fort.

Wenn sich der Generator beim Aufruf dieser Methode noch nicht bei einem yield-Ausdruck befindet, kann er zunächst zum ersten yield-Ausdruck vorrücken, bevor er den Wert sendet. Daher ist es bei einem PHP-Generator nicht nötig, ihn mit einem Generator::next()-Aufruf "vorzubereiten" (wie es bei Python der Fall ist).

Parameter-Liste

value

Der Wert, der an den Generator gesendet werden soll. Dieser Wert ist dann der Rückgabewert des yield-Ausdrucks, bei dem der Generator gerade steht.

Rückgabewerte

Gibt den gelieferten Wert zurück.

Beispiele

Beispiel #1 Werte mittels Generator::send() einfügen

<?php
function printer() {
echo
"I'm printer!".PHP_EOL;
while (
true) {
$string = yield;
echo
$string.PHP_EOL;
}
}

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

Das oben gezeigte Beispiel erzeugt folgende Ausgabe:

I'm printer!
Hello world!
Bye world!

add a note

User Contributed Notes 4 notes

up
74
sfroelich01 at sp dot gm dot ail dot am dot com
12 years ago
Reading the example, it is a bit difficult to understand what exactly to do with this. The example below is a simple example of what you can do this.<?phpfunction nums() {    for ($i = 0; $i < 5; ++$i) {                //get a value from the caller        $cmd = (yield $i);                if($cmd == 'stop')            return;//exit the function        }     }$gen = nums();foreach($gen as $v){    if($v == 3)//we are satisfied        $gen->send('stop');        echo "{$v}\n";}//Output0123?>
up
10
php at didatus dot de
4 years ago
If you want to use generator::send() within a foreach loop, you will most likely get an unexpected result. The Generator::send() method resumes the generator, which means the pointer within the generator is moved to the next element in the generator list.Here is an example:<?phpclass ApiDummy{    private static $apiDummyData = ['a', 'b', 'c', 'd', 'e'];    public static function getAll(): Generator {        foreach (self::$apiDummyData as $entry) {            echo 'yielding $elem' . PHP_EOL;            $newElem = (yield $entry);            echo 'yield return: ' . $newElem . PHP_EOL;        }    }}$generator = ApiDummy::getAll();// example iteration one with unexpected resultforeach ($generator as $elem) {    echo 'value from generator: ' . $elem . PHP_EOL;    $generator->send($elem . '+');}// example iteration two with the expected resultwhile ($generator->valid()) {    $elem = $generator->current();    echo 'value from generator: ' . $elem . PHP_EOL;    $generator->send($elem . '+');}?>The result of example iteration one:yielding $elemvalue from generator: ayield return: a+yielding $elemyield return:yielding $elemvalue from generator: cyield return: c+yielding $elemyield return:yielding $elemvalue from generator: eyield return: e+As you can see, the values b and d are not printed out and also not extended by the + sign.The foreach loop receives the first yield and the send call causes a second yield within the first loop. Therefor the second loop already receives the third yield and so on.To avoid this, one solution could be to use a while loop and the Generator::send() method to move the generator cursor forward and the Generator::current() method to retrieve the current value. The loop can be controlled with the Generator::valid() method which returns false, if the generator has finished. See example iterator two. The expected result of example iteration two:yielding $elemvalue from generator: ayield return: a+yielding $elemvalue from generator: byield return: b+yielding $elemvalue from generator: cyield return: c+yielding $elemvalue from generator: dyield return: d+yielding $elemvalue from generator: eyield return: e+
up
4
anonymous at example dot com
6 years ago
As of 7.3, the behavior of a generator in a foreach loop depends on whether or not it expects to receive data. Relevant if you are experiencing "skips".<?phpclass X implements IteratorAggregate {    public function getIterator(){        yield from [1,2,3,4,5];    }    public function getGenerator(){        foreach ($this as $j => $each){            echo "getGenerator(): yielding: {$j} => {$each}\n";            $val = (yield $j => $each);            yield; // ignore foreach's next()            echo "getGenerator(): received: {$j} => {$val}\n";        }    }}$x = new X;foreach ($x as $i => $val){    echo "getIterator(): {$i} => {$val}\n";}echo "\n";$gen = $x->getGenerator();foreach ($gen as $j => $val){    echo "getGenerator(): sending:  {$j} => {$val}\n";    $gen->send($val);}?>getIterator(): 0 => 1getIterator(): 1 => 2getIterator(): 2 => 3getIterator(): 3 => 4getIterator(): 4 => 5getGenerator(): yielding: 0 => 1getGenerator(): sending:  0 => 1getGenerator(): received: 0 => 1getGenerator(): yielding: 1 => 2getGenerator(): sending:  1 => 2getGenerator(): received: 1 => 2getGenerator(): yielding: 2 => 3getGenerator(): sending:  2 => 3getGenerator(): received: 2 => 3getGenerator(): yielding: 3 => 4getGenerator(): sending:  3 => 4getGenerator(): received: 3 => 4getGenerator(): yielding: 4 => 5getGenerator(): sending:  4 => 5getGenerator(): received: 4 => 5
up
11
sergei dot solomonov at gmail dot com
12 years ago
<?phpfunction foo() {    $string = yield;    echo $string;    for ($i = 1; $i <= 3; $i++) {        yield $i;    }}$generator = foo();$generator->send('Hello world!');foreach ($generator as $value) echo "$value\n";?>This code falls with the error:PHP Fatal error:  Uncaught exception 'Exception' with message 'Cannot rewind a generator that was already run'.foreach internally calls rewind, you should remember this!
To Top