我們來想一下,如果把集合對(duì)象和對(duì)集合對(duì)象的操作放在一起,當(dāng)我們想換一種方式遍歷集合對(duì)象中元素時(shí),就需要修改集合對(duì)象了,違背“單一職責(zé)原則”,而迭代器模式將數(shù)據(jù)結(jié)構(gòu)和數(shù)據(jù)結(jié)構(gòu)的算法分離開,兩者可獨(dú)立發(fā)展。
php迭代器(Iterator)的作用:
允許對(duì)象以自己的方式迭代內(nèi)部的數(shù)據(jù),從而使它可以被循環(huán)訪問,Iterator接口摘要如下:
Iterator extends Traversable { //返回當(dāng)前索引游標(biāo)指向的元素 abstract public mixed current ( void ) //返回當(dāng)前索引游標(biāo)指向的鍵 abstract public scalar key ( void ) //移動(dòng)當(dāng)前索引游標(biāo)到下一元素 abstract public void next ( void ) //重置索引游標(biāo) abstract public void rewind ( void ) //判斷當(dāng)前索引游標(biāo)指向的元素是否有效 abstract public boolean valid ( void ) }
迭代器優(yōu)缺點(diǎn)分析:
優(yōu)點(diǎn):
1.支持多種遍歷方式。比如有序列表,我們根據(jù)需要提供正序遍歷、倒序遍歷兩種迭代器。用戶只需要得到我們的迭代器,就可以對(duì)集合執(zhí)行遍歷操作
2.簡(jiǎn)化了聚合類。由于引入了迭代器,原有的集合對(duì)象不需要自行遍歷集合元素了
3.增加新的聚合類和迭代器類很方便,兩個(gè)維度上可各自獨(dú)立變化
4.為不同的集合結(jié)構(gòu)提供一個(gè)統(tǒng)一的接口,從而支持同樣的算法在不同的集合結(jié)構(gòu)上操作
缺點(diǎn):
迭代器模式將存儲(chǔ)數(shù)據(jù)和遍歷數(shù)據(jù)的職責(zé)分離增加新的集合對(duì)象時(shí)需要增加對(duì)應(yīng)的迭代器類,類的個(gè)數(shù)成對(duì)增加,在一定程度上增加系統(tǒng)復(fù)雜度。
下面是一個(gè)簡(jiǎn)單的例子演示Iterator的使用方法:
<?php /** * 該類允許外部迭代自己內(nèi)部私有屬性$_test,并演示迭代過程 * * @author 瘋狂老司機(jī) */ class TestIterator implements Iterator { /* * 定義要進(jìn)行迭代的數(shù)組 */ private $_test = array('dog', 'cat', 'pig'); /* * 索引游標(biāo) */ private $_key = 0; /* * 執(zhí)行步驟 */ private $_step = 0; /** * 將索引游標(biāo)指向初始位置 * * @see TestIterator::rewind() */ public function rewind() { echo '第'.++$this->_step.'步:執(zhí)行 '.__METHOD__.'<br>'; $this->_key = 0; } /** * 判斷當(dāng)前索引游標(biāo)指向的元素是否設(shè)置 * * @see TestIterator::valid() * @return bool */ public function valid() { echo '第'.++$this->_step.'步:執(zhí)行 '.__METHOD__.'<br>'; return isset($this->_test[$this->_key]); } /** * 將當(dāng)前索引指向下一位置 * * @see TestIterator::next() */ public function next() { echo '第'.++$this->_step.'步:執(zhí)行 '.__METHOD__.'<br>'; $this->_key++; } /** * 返回當(dāng)前索引游標(biāo)指向的元素的值 * * @see TestIterator::current() * @return value */ public function current() { echo '第'.++$this->_step.'步:執(zhí)行 '.__METHOD__.'<br>'; return $this->_test[$this->_key]; } /** * 返回當(dāng)前索引值 * * @return key * @see TestIterator::key() */ public function key() { echo '第'.++$this->_step.'步:執(zhí)行 '.__METHOD__.'<br>'; return $this->_key; } } $iterator = new TestIterator(); foreach($iterator as $key => $value){ echo "輸出索引為{$key}的元素".":$value".'<br><br>'; } ?>
以上例子將輸出:
第1步:執(zhí)行 TestIterator::rewind 第2步:執(zhí)行 TestIterator::valid 第3步:執(zhí)行 TestIterator::current 第4步:執(zhí)行 TestIterator::key 輸出索引為0的元素:dog 第5步:執(zhí)行 TestIterator::next 第6步:執(zhí)行 TestIterator::valid 第7步:執(zhí)行 TestIterator::current 第8步:執(zhí)行 TestIterator::key 輸出索引為1的元素:cat 第9步:執(zhí)行 TestIterator::next 第10步:執(zhí)行 TestIterator::valid 第11步:執(zhí)行 TestIterator::current 第12步:執(zhí)行 TestIterator::key 輸出索引為2的元素:pig 第13步:執(zhí)行 TestIterator::next 第14步:執(zhí)行 TestIterator::valid
從以上例子可以看出,如果執(zhí)行valid返回false,則循環(huán)就此結(jié)束。
相了解