溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

PHP中的Generator是什么

發(fā)布時(shí)間:2020-11-26 09:46:45 來源:億速云 閱讀:592 作者:栢白 欄目:編程語言

這篇文章主要介紹了PHP中的Generator是什么,具有一定借鑒價(jià)值,需要的朋友可以參考下。下面就和我一起來看看吧。

何為 Generator

從 PHP 5.5 開始,PHP 加入了一個(gè)新的特性,那就是 Generator,中文譯為生成器。生成器可以簡(jiǎn)單地用來實(shí)現(xiàn)對(duì)象的迭代,讓我們先從官方的一個(gè)小例子說起。

xrange

在 PHP 中,我們都知道,有一個(gè)函數(shù)叫做 range,用來生成一個(gè)等差數(shù)列的數(shù)組,然后我們可以用這個(gè)數(shù)組進(jìn)行 foreach 的迭代。具體就想這樣。

foreach (range(1, 100, 2) as $num) {
    echo $num . PHP_EOL;
}

這一段代碼就會(huì)輸出首項(xiàng)為 1,末項(xiàng)為 100,公差為 2 的等差數(shù)列。它的執(zhí)行順序是這樣的。首先,range(1, 100, 2) 會(huì)生成一個(gè)數(shù)組,里面存了上面那樣的一個(gè)等差數(shù)列,之后在 foreach 中對(duì)這個(gè)數(shù)組進(jìn)行迭代。

那么,這樣就會(huì)出現(xiàn)一個(gè)問題,如果我要生成 100 萬個(gè)數(shù)字呢?那我們就要占用上百兆內(nèi)存。雖然現(xiàn)在內(nèi)存很便宜,但是我們也不能這么浪費(fèi)內(nèi)存嘛。那么這時(shí),我們的生成器就可以排上用場(chǎng)了??紤]下面的代碼。

function xrange($start, $limit, $step = 1) {
    while ($start <= $limit) {
        yield $start;
        $start += $step;
    }
}

foreach (xrange(1, 100, 2) as $num) {
    echo $num . PHP_EOL;
}

這段代碼所的出來的結(jié)果,和前面的那段代碼一模一樣,但是,它內(nèi)部的原理是天翻地覆了。

我們剛才說了,前面的代碼,range 會(huì)生成一個(gè)數(shù)組,然后 foreach 來迭代這個(gè)數(shù)組,從而取出某一個(gè)值。但是這段代碼呢,我們重新定義了一個(gè) xrange 函數(shù),在函數(shù)中,我們用了一個(gè)關(guān)鍵字 yield。我們都知道定義一個(gè)函數(shù),希望它返回一個(gè)值得時(shí)候,用 return 來返回。那么這個(gè) yield 呢,也可以返回一個(gè)值,但是,它和 return 是截然不同的。

使用 yield 關(guān)鍵字,可以讓函數(shù)在運(yùn)行的時(shí)候,中斷,同時(shí)會(huì)保存整個(gè)函數(shù)的上下文,返回一個(gè) Generator 類型的對(duì)象。在執(zhí)行對(duì)象的 next 方法時(shí),會(huì)重新加載中斷時(shí)的上下文,繼續(xù)運(yùn)行,直到出現(xiàn)下一個(gè) yield 為止,如果后面沒有再出現(xiàn) yield,那么就認(rèn)為整個(gè)生成器結(jié)束了。

這樣,我們上面的函數(shù)調(diào)用可以等價(jià)地寫成這樣。

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

在這里,$num 是一個(gè) Generator 的對(duì)象。我們?cè)谶@里看到三個(gè)方法,validcurrentnext。當(dāng)我們函數(shù)執(zhí)行完了,后面沒有 yield 中斷了,那么我們?cè)?xrange 函數(shù)就執(zhí)行完了,那么 valid 方法就會(huì)變成 false。而 current 呢,會(huì)返回當(dāng)前 yield 后面的值,這是,生成器的函數(shù)會(huì)中斷。那么在調(diào)用 next 方法之后,函數(shù)會(huì)繼續(xù)執(zhí)行,直到下一個(gè) yield 出現(xiàn),或者函數(shù)結(jié)束。

好了,到這里,我們看到了通過 yield 來“生成”一個(gè)值并返回。其實(shí),yield 其實(shí)也可以這么寫 $ret = yield;。同返回值一樣,這里是將一個(gè)值在繼續(xù)執(zhí)行函數(shù)的時(shí)候,傳值進(jìn)函數(shù),可以通過 Generator::send($value) 來使用。例如。

function sum()
{
    $ret = yield;
    echo $ret . PHP_EOL;
}

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

這樣,程序就會(huì)打印出 send 方法傳進(jìn)去的字符串了。在 yield 的兩邊可以同時(shí)有調(diào)用。

function xrange($start, $limit, $step = 1) {
    while ($start <= $limit) {
        $ret = yield $start;
        $start += $step;
        echo $ret . PHP_EOL;
    }
}

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

而像這樣的使用,send() 可以返回下一個(gè) yield 的返回。

其它的 Generator 方法

Generator::key()

對(duì)于 yield,我們可以這樣使用 yield $id => $value,這是,我們可以通過 key 方法來獲取 $id,而 current 方法返回的是 $value

Generator::rewind()

這個(gè)方法,可以幫我們讓生成器重新開始執(zhí)行并保存上下文,同時(shí)呢,會(huì)返回第一個(gè) yield 返回的內(nèi)容。在第一次執(zhí)行 send 方法的時(shí)候,rewind 會(huì)被隱式調(diào)用。

Generator::throw()

這個(gè)方法,向生成器中,拋送一個(gè)異常。

后記

yield 作為 PHP 5.5 的新特性,讓我們用了新的方法來高效地迭代數(shù)據(jù)。同時(shí),我們還可以使用 yield 來實(shí)現(xiàn)協(xié)程。

以上就是PHP中的Generator是什么的詳細(xì)內(nèi)容了,看完之后是否有所收獲呢?如果想了解更多相關(guān)內(nèi)容,歡迎來億速云行業(yè)資訊!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI