PHP迭代器和生成器用法實例分析
本文實例講述了PHP迭代器和生成器用法。分享給大家供大家參考,具體如下:
迭代器
迭代器實際是一個實現(xiàn)了Iterator的類,可以用foreach進行遍歷。
例如:
<?php
class Sample implements Iterator{
private $curIndex=0;
private $items=null;
public function __construct($_items) {
$this->items = $_items;
}
public function current (){
echo "current\n";
return $this->items[$this->curIndex];
}
public function key (){
echo "key\n";
return $this->curIndex;
}
public function next (){
echo "next\n";
$this->curIndex++;
}
public function rewind (){
$this->curIndex = 0;
}
public function send ( $value ){
if($value == "stop"){
$this->curIndex = null;
}
}
public function valid (){
echo "valid\n";
return isset($this->items[$this->curIndex]);
}
}
$sample = new Sample([1,2,3]);
foreach ($sample as $k =>$v){
}
輸出
valid current key next
可以看到foreach 是先調(diào)用valid判斷迭代器是否有效,然后再調(diào)用current獲取當(dāng)前值,同時調(diào)用next移動key到指向下一個值(輸出key是因為 $k=>$v的緣故)。
生成器
讓我們先看一下官方文檔
生成器提供了一種更容易的方法來實現(xiàn)簡單的對象迭代,相比較定義類實現(xiàn) Iterator 接口的方式,性能開銷和復(fù)雜性大大降低。
生成器允許你在 foreach 代碼塊中寫代碼來迭代一組數(shù)據(jù)而不需要在內(nèi)存中創(chuàng)建一個數(shù)組, 那會使你的內(nèi)存達到上限,或者會占據(jù)可觀的處理時間。
相反,你可以寫一個生成器函數(shù),就像一個普通的自定義函數(shù)一樣, 和普通函數(shù)只返回一次不同的是, 生成器可以根據(jù)需要 yield 多次,以便生成需要迭代的值。
PHP 將會在每次需要值的時候調(diào)用生成器函數(shù),并在產(chǎn)生一個值之后保存生成器的狀態(tài),這樣它就可以在需要產(chǎn)生下一個值的時候恢復(fù)調(diào)用狀態(tài)。
下面是php官方文檔中的示例
<?php
function gen_one_to_three() {
for ($i = 1; $i <= 3; $i++) {
//注意變量$i的值在不同的yield之間是保持傳遞的。
yield $i;
}
}
$generator = gen_one_to_three();
foreach ($generator as $value) {
echo "$value\n";
}
var_dump($generator); //實際上是Generator對象
如上,若把3修改成10000,對于$generator實際上沒有區(qū)別,它只是保存了一個當(dāng)前值(當(dāng)然還有相關(guān)的內(nèi)部狀態(tài),這里是為了簡化),并沒有產(chǎn)生10000個數(shù)。
從中可以看出生成器的優(yōu)勢在于減少內(nèi)存的使用,在需要時才生成對應(yīng)的值。
查看php文檔,我們可以看到Generator實際也是Iterator的具體實現(xiàn),yield調(diào)用時就是返回的Generator對象。
那么怎么理解迭代器和生成器的關(guān)系呢?
其實,生成器是迭代器的實現(xiàn)+yield,產(chǎn)生了生成器對象。
我們也可以自己定義一個類似yield的函數(shù),如下:
function myYeild(){
$args = func_get_args();
return new Sample($args);
}
$generator = myYeild(1,2,3);
foreach ($generator as $value) {
echo "$value\n";
}
注意,我們的myYeild,是不能和php內(nèi)置的yeild那么使用的,因為yeild會保存調(diào)用上下文,臨時離開,并沒有return。
這里只是類比一下。
既然yeild可以把普通的對象包裝成generator,那么我們的iterator通過yeild也可以像Generator一樣嗎?
答案有點悲傷,yeild是把傳入的值作為參數(shù)生成Generator實例,它并不知道我們的iterator。不過這樣設(shè)計也是合理的,
以防我們自己的iterator不靠譜。
實際使用場合
- 數(shù)據(jù)庫遍歷
可以結(jié)合游標,遍歷數(shù)據(jù)庫時,不需要一次返回所有數(shù)據(jù),而是每次取一行。
class AllUser implements \Iterator
{
protected $index = 0;
protected $data = [];
public function __construct()
{
$link = mysqli_connect('192.168.0.91', 'root', '123', 'xxx');
$rec = mysqli_query($link, 'select id from doc_admin');
$this->data = mysqli_fetch_all($rec, MYSQLI_ASSOC);
}
//1 重置迭代器
public function rewind()
{
$this->index = 0;
}
//2 驗證迭代器是否有數(shù)據(jù)
public function valid()
{
return $this->index < count($this->data);
}
//3 獲取當(dāng)前內(nèi)容
public function current()
{
$id = $this->data[$this->index];
return User::find($id);
}
//4 移動key到下一個
public function next()
{
return $this->index++;
}
//5 迭代器位置key
public function key()
{
return $this->index;
}
}
//實現(xiàn)迭代遍歷用戶表
$users = new AllUser();
//可實時修改
foreach ($users as $user){
$user->add_time = time();
$user->save();
}
- 文件遍歷
一次讀取一行 - 實現(xiàn)Iterator接口,讓普通類可以使用foreach遍歷。
- 協(xié)程,參見鳥哥則這篇文章。
注意:可以在生成器的函數(shù)前加"&",可以使用引用。在函數(shù)里直接return會終止生成器。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php常用函數(shù)與技巧總結(jié)》、《php字符串(string)用法總結(jié)》、《PHP數(shù)組(Array)操作技巧大全》、《PHP數(shù)據(jù)結(jié)構(gòu)與算法教程》及《php程序設(shè)計算法總結(jié)》
希望本文所述對大家PHP程序設(shè)計有所幫助。
相關(guān)文章
PHP mail 通過Windows的SMTP發(fā)送郵件失敗的解決方案
今天調(diào)試WordPress的郵件發(fā)送功能,總是提示:SMTP server response: 501 5.5.4 Invalid Address。用telnet測試SMTP是沒有任何問題的2009-05-05
php中file_get_contents與curl性能比較分析
這篇文章主要介紹了php中file_get_contents與curl性能比較,以實例形式詳細分析了file_get_contents與curl的區(qū)別以及運行效率的對比,需要的朋友可以參考下2014-11-11
PHP+JS實現(xiàn)批量刪除數(shù)據(jù)功能示例
這篇文章主要介紹了PHP+JS實現(xiàn)批量刪除數(shù)據(jù)功能,結(jié)合實例形式分析了php結(jié)合js控制頁面元素的選中與提交,以及php操作mysql實現(xiàn)批量刪除功能的相關(guān)實現(xiàn)技巧,末尾還附帶了一個php數(shù)據(jù)庫操作類,需要的朋友可以參考下2017-11-11

