PHP中什么是生成器(Generator)?怎么使用?下面本篇文章帶大家深入講解一下PHP 中的生成器,希望對(duì)大家有所幫助!
談到駕駛,速度并非一切。但在網(wǎng)絡(luò)上,速度變得與眾不同。你的應(yīng)用程序越快,用戶體驗(yàn)越好。這篇文章是關(guān)于 PHP 生成器的,那么我們?yōu)槭裁匆懻撍俣饶??你很快就?huì)發(fā)現(xiàn),生成器在速度和內(nèi)存的管理上發(fā)揮著巨大的作用。
PHP生成器是什么?
生成器是在 PHP 5.5 版本中添加的,它提供了一種簡(jiǎn)單的方法來(lái)遍歷數(shù)據(jù),而不需要在內(nèi)存中構(gòu)建數(shù)組。是不是有點(diǎn)疑惑?那舉一個(gè)例子,展示使用生成器是一個(gè)好方式。
首先,創(chuàng)建一個(gè) generator.php 文件,它將貫穿我們整個(gè)例子。創(chuàng)建文件之后,我們添加一段代碼。
<?php function getRange ($max = 10) { $array = []; for ($i = 1; $i < $max; $i++) { $array[] = $i; } return $array; } foreach (getRange(15) as $range) { echo "Dataset {$range} <br>"; }
我們可以在創(chuàng)建 generator.php 文件所在目錄中快速啟動(dòng)一個(gè)內(nèi)置的 PHP 服務(wù)器:
php -S localhost:8000
如果用瀏覽器打開(kāi) http://localhost:8000/generator.php ,我們應(yīng)該看到這樣的結(jié)果:
這段代碼的自解釋性并不是太好. 讓我們稍微改動(dòng)一下代碼
<?php foreach (getRange(PHP_INT_MAX) as $range) { echo "Dataset {$range} <br>"; }
現(xiàn)在, 上面的這段代碼能夠生成的最大值是 PHP_INT_MAX (也就是 PHP 能夠生成的最大值). 當(dāng)我們這樣修改后刷新瀏覽器,我們注意到這次有一些不一樣. 這段生成器腳本拋出了一條 warning 信息 .
有點(diǎn)遺憾的是 PHP 耗盡了內(nèi)存. 你能夠想到的解決方法可能包括增加 php.ini 文件中 memory_limit 的上限. 不過(guò)平心而論,這個(gè)腳本既不高效又占用內(nèi)存, 我們需要的是一個(gè)高效且占用內(nèi)存低的腳本。
使用生成器
讓我們?cè)谏厦娑x相同的函數(shù),用相同的值 PHP_INT_MAX 調(diào)用它,然后再次運(yùn)行。但是這一次我們將創(chuàng)建一個(gè)生成器函數(shù)。
<?php function getRange ($max = 10) { for ($i = 1; $i < $max; $i++) { yield $i; } } foreach (getRange(PHP_INT_MAX) as $range) { echo "Dataset {$range} <br>"; }
解析 getRange
函數(shù),這次,我們只循環(huán)遍歷值和 yield
輸出。 yield
與返回值類似,因?yàn)樗彩菑暮瘮?shù)返回一個(gè)值,但唯一的區(qū)別是 yield
只會(huì)在需要時(shí)返回一個(gè)值,并且不會(huì)嘗試將整個(gè)數(shù)據(jù)集保留在內(nèi)存中。
如果您轉(zhuǎn)到瀏覽器,您應(yīng)該會(huì)看到頁(yè)面上顯示的數(shù)據(jù)。給定適當(dāng)?shù)臅r(shí)間,瀏覽器最終顯示數(shù)據(jù)。
注意: 生成器只能在函數(shù)中使用。
為什么要使用生成器
有時(shí)候,我們可能會(huì)遇到想要解析一個(gè)龐大的數(shù)據(jù)集(也可能是日志文件),也可能對(duì)一個(gè)大型數(shù)據(jù)庫(kù)的結(jié)果集執(zhí)行計(jì)算,等等情況。我們不想讓這些數(shù)據(jù)全部加載到內(nèi)存中。我們應(yīng)該盡可能的保存相應(yīng)的內(nèi)存狀態(tài)。數(shù)據(jù)不一定要很大——無(wú)論數(shù)據(jù)有多小,生成器都是有效的。別忘了,我們的目的是使用更少的內(nèi)存來(lái)盡可能快的處理數(shù)據(jù)。
返回鍵值對(duì)
有時(shí)候,我們的數(shù)據(jù)是基于 key-value 時(shí)才更有說(shuō)服力。使用生成器時(shí),我們可能會(huì)生成下面這樣的鍵值對(duì)。
<?php function getRange ($max = 10) { for ($i = 1; $i < $max; $i++) { $value = $i * mt_rand(); yield $i => $value; } }
然后,我們可以使用這個(gè)鍵值對(duì),就像使用任意的數(shù)組一樣。
<?php foreach (getRange(PHP_INT_MAX) as $range => $value) { echo "Dataset {$range} has {$value} value<br>"; }
傳遞參數(shù)到生成器中
生成器也能接收傳參。這意味這生成器允許我們向其中注入?yún)?shù),作為一個(gè)命令或者其他作用。例如,我們向生成器發(fā)送一個(gè)值,讓它停止執(zhí)行或者修改輸出結(jié)果。使用上面的 getRange
函數(shù),我們可以實(shí)現(xiàn)這一點(diǎn)。
<?php function getRange ($max = 10) { for ($i = 1; $i < $max; $i++) { $injected = yield $i; if ($injected === 'stop') return; } }
要發(fā)送注入這個(gè)值,我們可以這樣做。
<?php $generator = getRange(PHP_INT_MAX); foreach ($generator as $range) { if ($range === 10000) { $generator->send('stop'); } echo "Dataset {$range} <br>"; }
注意: 在生成器中使用 return ,會(huì)跳出生成器。
不要濫用生成器
雖然使用 PHP_INT_MAX 有點(diǎn)過(guò)了. 但對(duì)我來(lái)說(shuō), PHP_INT_MAX 即 2147483647 也就是:
二十億四千七百四十萬(wàn)四千八萬(wàn)三千六百四十七
生成器使內(nèi)存使用更高效。但如果濫用,一樣會(huì)造成內(nèi)存相關(guān)的問(wèn)題。
總結(jié)
生成器提供了難以忽視的顯著性的能提升。大多數(shù)的時(shí)候,我們不需要高配置的服務(wù)器來(lái)運(yùn)行代碼。我們只需要做一點(diǎn)重構(gòu),生成器是非常有用的,我們應(yīng)該多多使用它們
英文原文地址:https://scotch.io/tutorials/understanding-php-generators
推薦學(xué)習(xí):《PHP視頻教程》