摘要:在中,我們都知道,有一個函數(shù)叫做,用來生成一個等差數(shù)列的數(shù)組,然后我們可以用這個數(shù)組進(jìn)行的迭代。這一段代碼就會輸出首項(xiàng)為,末項(xiàng)為,公差為的等差數(shù)列。它的執(zhí)行順序是這樣的。
何為 Generator
從 PHP 5.5 開始,PHP 加入了一個新的特性,那就是 Generator,中文譯為生成器。生成器可以簡單地用來實(shí)現(xiàn)對象的迭代,讓我們先從官方的一個小例子說起。
xrange在 PHP 中,我們都知道,有一個函數(shù)叫做 range,用來生成一個等差數(shù)列的數(shù)組,然后我們可以用這個數(shù)組進(jìn)行 foreach 的迭代。具體就想這樣。
foreach (range(1, 100, 2) as $num) { echo $num . PHP_EOL; }
這一段代碼就會輸出首項(xiàng)為 1,末項(xiàng)為 100,公差為 2 的等差數(shù)列。它的執(zhí)行順序是這樣的。首先,range(1, 100, 2) 會生成一個數(shù)組,里面存了上面那樣的一個等差數(shù)列,之后在 foreach 中對這個數(shù)組進(jìn)行迭代。
那么,這樣就會出現(xiàn)一個問題,如果我要生成 100 萬個數(shù)字呢?那我們就要占用上百兆內(nèi)存。雖然現(xiàn)在內(nèi)存很便宜,但是我們也不能這么浪費(fèi)內(nèi)存嘛。那么這時,我們的生成器就可以排上用場了。考慮下面的代碼。
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 會生成一個數(shù)組,然后 foreach 來迭代這個數(shù)組,從而取出某一個值。但是這段代碼呢,我們重新定義了一個 xrange 函數(shù),在函數(shù)中,我們用了一個關(guān)鍵字 yield。我們都知道定義一個函數(shù),希望它返回一個值得時候,用 return 來返回。那么這個 yield 呢,也可以返回一個值,但是,它和 return 是截然不同的。
使用 yield 關(guān)鍵字,可以讓函數(shù)在運(yùn)行的時候,中斷,同時會保存整個函數(shù)的上下文,返回一個 Generator 類型的對象。在執(zhí)行對象的 next 方法時,會重新加載中斷時的上下文,繼續(xù)運(yùn)行,直到出現(xiàn)下一個 yield 為止,如果后面沒有再出現(xiàn) yield,那么就認(rèn)為整個生成器結(jié)束了。
這樣,我們上面的函數(shù)調(diào)用可以等價地寫成這樣。
$nums = xrange(1, 100, 2); while ($nums->valid()) { echo $nums->current() . " "; $nums->next(); }
在這里,$num 是一個 Generator 的對象。我們在這里看到三個方法,valid、current 和 next。當(dāng)我們函數(shù)執(zhí)行完了,后面沒有 yield 中斷了,那么我們在 xrange 函數(shù)就執(zhí)行完了,那么 valid 方法就會變成 false。而 current 呢,會返回當(dāng)前 yield 后面的值,這是,生成器的函數(shù)會中斷。那么在調(diào)用 next 方法之后,函數(shù)會繼續(xù)執(zhí)行,直到下一個 yield 出現(xiàn),或者函數(shù)結(jié)束。
好了,到這里,我們看到了通過 yield 來“生成”一個值并返回。其實(shí),yield 其實(shí)也可以這么寫 $ret = yield;。同返回值一樣,這里是將一個值在繼續(xù)執(zhí)行函數(shù)的時候,傳值進(jìn)函數(shù),可以通過 Generator::send($value) 來使用。例如。
function sum() { $ret = yield; echo $ret . PHP_EOL; } $sum = sum(); $sum->send("I am from outside.");
這樣,程序就會打印出 send 方法傳進(jìn)去的字符串了。在 yield 的兩邊可以同時有調(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() . " "; $nums->send($nums->current() + 1); }
而像這樣的使用,send() 可以返回下一個 yield 的返回。
其它的 Generator 方法 Generator::key()對于 yield,我們可以這樣使用 yield $id => $value,這是,我們可以通過 key 方法來獲取 $id,而 current 方法返回的是 $value。
Generator::rewind()這個方法,可以幫我們讓生成器重新開始執(zhí)行并保存上下文,同時呢,會返回第一個 yield 返回的內(nèi)容。在第一次執(zhí)行 send 方法的時候,rewind 會被隱式調(diào)用。
Generator::throw()這個方法,向生成器中,拋送一個異常。
后記yield 作為 PHP 5.5 的新特性,讓我們用了新的方法來高效地迭代數(shù)據(jù)。同時,我們還可以使用 yield 來實(shí)現(xiàn)協(xié)程。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/23089.html
摘要:別名相當(dāng)于前面先引入了路由插件,然后顯式聲明要用路由。注意到,等都是頁面也可以是組件,接著注冊路由器,然后開始配置路由。 搭建環(huán)境 工欲善其事必先利其器,我們的學(xué)習(xí)計(jì)劃從學(xué)會搭建Vue所需要的環(huán)境開始,node和npm的環(huán)境不用說是必須的,現(xiàn)在前端流程化很熱門,基本上新的技術(shù)都會在這套流程的基礎(chǔ)上做開發(fā),我們只需要站在巨人的XX上裝*就可以了。我假設(shè)你的機(jī)子上已經(jīng)有了最新的node和n...
摘要:說到中的生成器,有人可能會想到協(xié)程,這里我們先不說如何實(shí)現(xiàn)協(xié)程,我們探究下的執(zhí)行過程。如果函數(shù)包含了關(guān)鍵字的,那么函數(shù)執(zhí)行后的返回值永遠(yuǎn)都是一個對象。如果函數(shù)內(nèi)部同事包含和該函數(shù)的返回值依然是對象,但是在生成對象時,語句后的代碼被忽略。 說到php中的Generator(生成器),有人可能會想到協(xié)程,這里我們先不說php如何實(shí)現(xiàn)協(xié)程,我們探究下Generator的執(zhí)行過程。 Gene...
摘要:從最開始的到封裝后的都在試圖解決異步編程過程中的問題。為了讓編程更美好,我們就需要引入來降低異步編程的復(fù)雜性。異步編程入門的全稱是前端經(jīng)典面試題從輸入到頁面加載發(fā)生了什么這是一篇開發(fā)的科普類文章,涉及到優(yōu)化等多個方面。 TypeScript 入門教程 從 JavaScript 程序員的角度總結(jié)思考,循序漸進(jìn)的理解 TypeScript。 網(wǎng)絡(luò)基礎(chǔ)知識之 HTTP 協(xié)議 詳細(xì)介紹 HTT...
閱讀 1121·2023-04-26 02:46
閱讀 624·2023-04-25 19:38
閱讀 639·2021-10-14 09:42
閱讀 1234·2021-09-08 09:36
閱讀 1354·2019-08-30 15:44
閱讀 1319·2019-08-29 17:23
閱讀 2237·2019-08-29 15:27
閱讀 801·2019-08-29 14:15