談到駕駛,速度並非一切。但在網路上,速度變得與眾不同。你的應用程式越快,用戶體驗越好。這篇文章是關於 PHP 生成器的,那我們為什麼要討論速度呢?你很快就會發現,生成器在速度和記憶體的管理上發揮著巨大的作用。
PHP 生成器是什麼?
生成器是在 PHP 5.5 版本中添加的,它提供了一種簡單的方法來遍歷數據,而不需要在記憶體中建立數組。是不是有點疑惑?那舉一個例子,展示使用生成器是一個好方式。
首先,建立一個 generator.php 文件,它將貫穿我們整個範例。創建文件之後,我們添加一段程式碼。
<?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>"; }
我們可以在建立generator.php 檔案所在目錄中快速啟動一個內建的PHP 伺服器:
php -S localhost:8000
如果用瀏覽器開啟http://localhost:8000/generator.php ,我們應該要看到這樣的結果:
這段程式碼的自解釋性並不是太好。讓我們稍微改動一下程式碼
<?php foreach (getRange(PHP_INT_MAX) as $range) { echo "Dataset {$range} <br>"; }
現在,上面的這段程式碼能夠產生的最大值是PHP_INT_MAX (也就是PHP 能夠產生的最大值). 當我們這樣修改後刷新瀏覽器,我們注意到這次有些不一樣。這段生成器腳本拋出了一條 warning 資訊 .
有點遺憾的是 PHP 耗盡了記憶體。你能夠想到的解決方法可能包括增加 php.ini 檔案中 memory_limit 的上限。不過平心而論,這個腳本既不高效又佔用內存,我們需要的是一個高效且佔用內存低的腳本。
使用生成器
讓我們在上面定義相同的函數,用相同的值 PHP_INT_MAX 呼叫它,然後再次運行。但是這次我們將創建一個生成器函數。
<?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 函數,這次,我們只循環遍歷值和 yield 輸出。 yield 與傳回值類似,因為它也是從函數傳回一個值,但唯一的區別是 yield 只會在需要時傳回一個值,並且不會嘗試將整個資料集保留在記憶體中。
如果您轉到瀏覽器,您應該會看到頁面上顯示的資料。給定適當的時間,瀏覽器最終顯示資料。
注意: 生成器只能在函數中使用。
為什麼要使用生成器
有時候,我們可能會遇到想要解析一個龐大的資料集(也可能是日誌檔案),也可能對一個大型資料庫的結果集執行計算,等等情況。我們不想讓這些資料全部載入到記憶體中。我們應該盡可能的保存相應的記憶體狀態。數據不一定要很大 —— 無論數據有多小,生成器都是有效的。別忘了,我們的目的是使用更少的記憶體來盡可能快的處理資料。
回傳鍵值對
有時候,我們的資料是基於 key-value 時才更有說服力。使用生成器時,我們可能會產生下面這樣的鍵值對。
<?php function getRange ($max = 10) { for ($i = 1; $i < $max; $i++) { $value = $i * mt_rand(); yield $i => $value; } }
然後,我們可以使用這個鍵值對,就像使用任意的陣列一樣。
<?php foreach (getRange(PHP_INT_MAX) as $range => $value) { echo "Dataset {$range} has {$value} value<br>"; }
傳遞參數到生成器中
生成器也能接收傳參。這意味著這生成器允許我們向其中註入參數,作為一個命令或其他作用。例如,我們向生成器發送一個值,讓它停止執行或修改輸出結果。使用上面的 getRange 函數,我們可以實現這一點。
<?php function getRange ($max = 10) { for ($i = 1; $i < $max; $i++) { $injected = yield $i; if ($injected === 'stop') return; } }
要傳送注入這個值,我們可以這樣做。
<?php $generator = getRange(PHP_INT_MAX); foreach ($generator as $range) { if ($range === 10000) { $generator->send('stop'); } echo "Dataset {$range} <br>"; }
注意: 在生成器中使用 return ,會跳出生成器。
不要濫用生成器
雖然使用 PHP_INT_MAX 有點過了。但對我來說, PHP_INT_MAX 即2147483647 也就是:
二十億四千七百四十萬四千八萬三千六百四十七
生成器讓記憶體使用更有效率。但如果濫用,一樣會造成記憶體相關的問題。
總結
生成器提供了難以忽視的顯著性的能提升。大多數的時候,我們不需要高配置的伺服器來運行程式碼。我們只需要做一點重構,生成器是非常有用的,我們應該多使用它們。
以上是PHP 中的生成器(Generator)詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!