本文實(shí)例講述了PHP迭代器和生成器用法。分享給大家供大家參考,具體如下:
迭代器
迭代器實(shí)際是一個(gè)實(shí)現(xiàn)了Iterator的類,可以用foreach進(jìn)行遍歷。
例如:
?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)前值,同時(shí)調(diào)用next移動(dòng)key到指向下一個(gè)值(輸出key是因?yàn)?$k=>$v的緣故)。
生成器
讓我們先看一下官方文檔
生成器提供了一種更容易的方法來(lái)實(shí)現(xiàn)簡(jiǎn)單的對(duì)象迭代,相比較定義類實(shí)現(xiàn) Iterator 接口的方式,性能開(kāi)銷和復(fù)雜性大大降低。
生成器允許你在 foreach 代碼塊中寫代碼來(lái)迭代一組數(shù)據(jù)而不需要在內(nèi)存中創(chuàng)建一個(gè)數(shù)組, 那會(huì)使你的內(nèi)存達(dá)到上限,或者會(huì)占據(jù)可觀的處理時(shí)間。
相反,你可以寫一個(gè)生成器函數(shù),就像一個(gè)普通的自定義函數(shù)一樣, 和普通函數(shù)只返回一次不同的是, 生成器可以根據(jù)需要 yield 多次,以便生成需要迭代的值。
PHP 將會(huì)在每次需要值的時(shí)候調(diào)用生成器函數(shù),并在產(chǎn)生一個(gè)值之后保存生成器的狀態(tài),這樣它就可以在需要產(chǎn)生下一個(gè)值的時(shí)候恢復(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); //實(shí)際上是Generator對(duì)象
如上,若把3修改成10000,對(duì)于$generator實(shí)際上沒(méi)有區(qū)別,它只是保存了一個(gè)當(dāng)前值(當(dāng)然還有相關(guān)的內(nèi)部狀態(tài),這里是為了簡(jiǎn)化),并沒(méi)有產(chǎn)生10000個(gè)數(shù)。
從中可以看出生成器的優(yōu)勢(shì)在于減少內(nèi)存的使用,在需要時(shí)才生成對(duì)應(yīng)的值。
查看php文檔,我們可以看到Generator實(shí)際也是Iterator的具體實(shí)現(xiàn),yield調(diào)用時(shí)就是返回的Generator對(duì)象。
那么怎么理解迭代器和生成器的關(guān)系呢?
其實(shí),生成器是迭代器的實(shí)現(xiàn)+yield,產(chǎn)生了生成器對(duì)象。
我們也可以自己定義一個(gè)類似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那么使用的,因?yàn)閥eild會(huì)保存調(diào)用上下文,臨時(shí)離開(kāi),并沒(méi)有return。
這里只是類比一下。
既然yeild可以把普通的對(duì)象包裝成generator,那么我們的iterator通過(guò)yeild也可以像Generator一樣嗎?
答案有點(diǎn)悲傷,yeild是把傳入的值作為參數(shù)生成Generator實(shí)例,它并不知道我們的iterator。不過(guò)這樣設(shè)計(jì)也是合理的,
以防我們自己的iterator不靠譜。
實(shí)際使用場(chǎng)合
可以結(jié)合游標(biāo),遍歷數(shù)據(jù)庫(kù)時(shí),不需要一次返回所有數(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 驗(yàn)證迭代器是否有數(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 移動(dòng)key到下一個(gè) public function next() { return $this->index++; } //5 迭代器位置key public function key() { return $this->index; } } //實(shí)現(xiàn)迭代遍歷用戶表 $users = new AllUser(); //可實(shí)時(shí)修改 foreach ($users as $user){ $user->add_time = time(); $user->save(); }
注意:可以在生成器的函數(shù)前加"",可以使用引用。在函數(shù)里直接return會(huì)終止生成器。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php常用函數(shù)與技巧總結(jié)》、《php字符串(string)用法總結(jié)》、《PHP數(shù)組(Array)操作技巧大全》、《PHP數(shù)據(jù)結(jié)構(gòu)與算法教程》及《php程序設(shè)計(jì)算法總結(jié)》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
標(biāo)簽:迪慶 鷹潭 赤峰 烏魯木齊 山南 濟(jì)寧 三亞 南寧
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《PHP迭代器和生成器用法實(shí)例分析》,本文關(guān)鍵詞 PHP,迭代,器,和,生,成器,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。