欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP中迭代器的简单实现及Yii框架中的迭代器实现方法示例

程序员文章站 2022-06-17 14:44:03
本文实例讲述了php中迭代器的简单实现及yii框架中的迭代器实现方法。分享给大家供大家参考,具体如下:在*中我们可以看到其定义如下:迭代器有时又称光标(cursor)是程式设计的软件设计模式,可...

本文实例讲述了php中迭代器的简单实现及yii框架中的迭代器实现方法。分享给大家供大家参考,具体如下:

在*中我们可以看到其定义如下:

迭代器有时又称光标(cursor)是程式设计的软件设计模式,可在容器物件(container,例如list或vector)上遍访的接口,设计人员无需关心容器物件的内容。

各种语言实作iterator的方式皆不尽同,有些面向对象语言像java, c#, python, delphi都已将iterator的特性内建语言当中,完美的跟语言整合,我们称之隐式迭代器(implicit iterator),但像是c++语言本身就没有iterator的特色,但stl仍利用template实作了功能强大的iterator。

iterator另一方面还可以整合generator。有些语言将二者视为同一接口,有些语言则将之独立化。
地址:http://zh.wikipedia.org/zh-cn/%e8%bf%ad%e4%bb%a3%e5%99%a8

【iterator的简单实现】

/**
* iterator模式的简单实现类
*/
class sample implements iterator {
  private $_items ;
 
  public function __construct(&$data) {
    $this->_items = $data;
  }
  public function current() {
    return current($this->_items);
  }
 
  public function next() {
    next($this->_items);  
  }
 
  public function key() {
    return key($this->_items);
  }
 
  public function rewind() {
    reset($this->_items);
  }
 
  public function valid() {
    return ($this->current() !== false);
  }
}
 
/** demo */
$data = array(1, 2, 3, 4, 5);
$sa = new sample($data);
foreach ($sa as $key => $row) {
  echo $key, ' ', $row, '<br />';
}

在next()方法的实现时有过纠结,一直以为这里需要返回下一个的值,

这是因为一直以为这里的next就是next函数的实现,但是非也

在手册中我们可以看到其定义为

abstract public void iterator::next ( void )

其返回值类型为void

所以这里我们调用next函数就可以了,没有必要返回

另外,以上实现对于如下的数组是存在的问题

$data = array('0' => 11, "" => 22, 's3' => 33, 0, 0, "", false, 0, 1);

运行结果是输出:

0 11
22
s3 33
1 0
2 0
3

false后面的值就没有迭代显示出来了,具体原因还不清楚,留作下回分解

在yii框架中也有实现迭代器,它的实现避免了这个问题。

【yii框架中的迭代器实现】

在yii框架中的我们可以看到其迭代器的实现

在collections目录下的cmapiterator.php文件中,其实现如下:

class cmapiterator implements iterator {
/**
* @var array the data to be iterated through
*/
  private $_d;
/**
* @var array list of keys in the map
*/
  private $_keys;
/**
* @var mixed current key
*/
  private $_key;
 
/**
* constructor.
* @param array the data to be iterated through
*/
  public function __construct(&$data) {
    $this->_d=&$data;
    $this->_keys=array_keys($data);
  }
 
/**
* rewinds internal array pointer.
* this method is required by the interface iterator.
*/
  public function rewind() {                                         
    $this->_key=reset($this->_keys);
  }
 
/**
* returns the key of the current array element.
* this method is required by the interface iterator.
* @return mixed the key of the current array element
*/
  public function key() {
    return $this->_key;
  }
 
/**
* returns the current array element.
* this method is required by the interface iterator.
* @return mixed the current array element
*/
  public function current() {
    return $this->_d[$this->_key];
  }
 
/**
* moves the internal pointer to the next array element.
* this method is required by the interface iterator.
*/
  public function next() {
    $this->_key=next($this->_keys);
  }
 
/**
* returns whether there is an element at current position.
* this method is required by the interface iterator.
* @return boolean
*/
  public function valid() {
    return $this->_key!==false;
  }
}
 
$data = array('s1' => 11, 's2' => 22, 's3' => 33);
$it = new cmapiterator($data);
foreach ($it as $row) {
  echo $row, '<br />';
}

这与之前的简单实现相比,其位置的变化是通过控制key来实现的,这种实现的作用是为了避免false作为数组值时无法迭代