|
迭代器(Iterator)模式,它在一個(gè)很常見的過程上提供了一個(gè)抽象:位于對(duì)象圖不明部分的一組對(duì)象(或標(biāo)量)集合上的迭代。迭代有幾種不同的具體執(zhí)行方法:在數(shù)組屬性,集合對(duì)象,數(shù)組,甚至一個(gè)查詢結(jié)果集之上迭代。
在對(duì)象的世界里,迭代器模式要維持類似數(shù)組的功能,看作是一個(gè)非侵入性對(duì)象刻面(facet),Client類往往分離自真實(shí)對(duì)象實(shí)現(xiàn),指iterator接口。只要有可能,我們可以給迭代器傳送一個(gè)引用,代替將來可能發(fā)生變化的具體或抽象類。
參與者:
◆客戶端(Client):引用迭代器模式的方法在一組值或?qū)ο笊蠄?zhí)行一個(gè)循環(huán)。
◆迭代器(Iterator):在迭代過程上的抽象,包括next(),isFinished(),current()等方法。
◆具體迭代器(ConcreteIterators):在一個(gè)特定的對(duì)象集,如數(shù)組,樹,組合,集合等上實(shí)現(xiàn)迭代。
通過Traversable接口,php原生態(tài)支持迭代器模式,這個(gè)接口由Iterator和IteratorAggregate做了擴(kuò)展,這兩個(gè)子接口不僅是定義了一套標(biāo)準(zhǔn)的方法,每個(gè)Traversable對(duì)象都可以原封不動(dòng)地傳遞給foreach(),foreach是迭代器的主要客戶端,Iterator實(shí)現(xiàn)是真正的迭代器,而IteratorAggregate是有其它職責(zé)的Traversable對(duì)象,它通過getIterator()方法返回一個(gè)Iterator。
標(biāo)準(zhǔn)php庫是php中綁定的唯一通用目的面向?qū)ο髱欤x了額外的接口和公用類。OuterIterator實(shí)現(xiàn)裝飾一個(gè)Iterator,CachingIterator和LimitIterator是這個(gè)接口的兩個(gè)例子。
RecursiveIterator是Iterator接口為樹形結(jié)構(gòu)實(shí)現(xiàn)的一個(gè)擴(kuò)展,它定義了一組額外的方法檢查迭代中當(dāng)前元素的子對(duì)象是否存在。RecursiveArrayIterator和RecursiveDirectoryIterator是這個(gè)接口的實(shí)現(xiàn)示例,這些類型的迭代器可以原樣使用,或是用一個(gè)RecursiveIteratorIterator橋接到一個(gè)普通的迭代器契約。這個(gè)OuterIterator實(shí)現(xiàn)將會(huì)根據(jù)構(gòu)造參數(shù)執(zhí)行深度優(yōu)先或廣度優(yōu)先遍歷。
使用RecursiveIteratorIterator時(shí),可以將其傳遞給foreach,請(qǐng)看后面的代碼示例,了解RecursiveIterators的不同用法和它們的超集Iterator。最后,SeekableIterators向契約添加了一個(gè)seek()方法,它可以用于移動(dòng)Iterator的內(nèi)部狀態(tài)到一個(gè)特定的迭代點(diǎn)。
注意,迭代器是比對(duì)象集更好的抽象,因?yàn)槲覀兛梢宰孖nfiniteIterators,NoRewindIterators等,不用與普通數(shù)組陣列一致,因此,Iterator缺少count()函數(shù)等功能。
在php官方手冊(cè)中可以找到完整的SPL迭代器列表。得益于對(duì)php的強(qiáng)力支持,使用迭代器模式的大部分工作都包括在標(biāo)準(zhǔn)實(shí)現(xiàn)中,下面的代碼示例就利用了標(biāo)準(zhǔn)Iterator和RecursiveIterators的功能。
復(fù)制代碼 代碼如下:
<?php
/**
* Collection that wraps a numeric array.
* All five public methods are needed to implement
* the Iterator interface.
*/
class Collection implements Iterator
{
private $_content;
private $_index = 0;
public function __construct(array $content)
{
$this->_content = $content;
}
public function rewind()
{
$this->_index = 0;
}
public function valid()
{
return isset($this->_content[$this->_index]);
}
public function current()
{
return $this->_content[$this->_index];
}
public function key()
{
return $this->_index;
}
public function next()
{
$this->_index++;
}
}
$array = array('A', 'B', 'C', 'D');
echo "Collection: ";
foreach (new Collection($array) as $key => $value) {
echo "$key => $value. ";
}
echo "/n";
/**
* Usually IteratorAggregate is the interface to implement.
* It has only one method, which must return an Iterator
* already defined as another class (e.g. ArrayIterator)
* Iterator gives a finer control over the algorithm,
* because all the hook points of Iterator' contract
* are available for implementation.
*/
class NumbersSet implements IteratorAggregate
{
private $_content;
public function __construct(array $content)
{
$this->_content = $content;
}
public function contains($number)
{
return in_array($number, $this->_content);
}
/**
* Only this method is necessary to implement IteratorAggregate.
* @return Iterator
*/
public function getIterator()
{
return new ArrayIterator($this->_content);
}
}
echo "NumbersSet: ";
foreach (new NumbersSet($array) as $key => $value) {
echo "$key => $value. ";
}
echo "/n";
// let's play with RecursiveIterator implementations
$it = new RecursiveArrayIterator(array(
'A',
'B',
array(
'C',
'D'
),
array(
array(
'E',
'F'
),
array(
'G',
'H',
'I'
)
)
));
// $it is a RecursiveIterator but also an Iterator,
// so it loops normally over the four elements
// of the array.
echo "Foreach over a RecursiveIterator: ";
foreach ($it as $value) {
echo $value;
// but RecursiveIterators specify additional
// methods to explore children nodes
$children = $it->hasChildren() ? '{Yes}' : '{No}';
echo $children, ' ';
}
echo "/n";
// we can bridge it to a different contract via
// a RecursiveIteratorIterator, whose cryptic name
// should be read as 'an Iterator that spans over
// a RecursiveIterator'.
echo "Foreach over a RecursiveIteratorIterator: ";
foreach (new RecursiveIteratorIterator($it) as $value) {
echo $value;
}
echo "/n";
php技術(shù):PHP設(shè)計(jì)模式之迭代器模式的深入解析,轉(zhuǎn)載需保留來源!
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。