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