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