php中如何使对象可以像数组一样进行foreach循环
程序员文章站
2023-11-22 21:35:16
刚接触到题的时候,我也没有考虑到iterator模式,试了几个一般想法,失败以后。。。。就直接去翻看了foreach的源码实现,期望发现foreach处理对象的时候是否有什...
刚接触到题的时候,我也没有考虑到iterator模式,试了几个一般想法,失败以后。。。。就直接去翻看了foreach的源码实现,期望发现foreach处理对象的时候是否有什么特殊性,可以做为突破口。
跟踪了半天以后发现了核心逻辑中的一个奇怪的switch:
复制代码 代码如下:
switch (zend_iterator_unwrap(array, &iter tsrmls_cc)) {
default:
case zend_iter_invalid:
.....
break
case zend_iter_plain_object: {
......
break;
case zend_iter_plain_array:
.....
break;
case zend_iter_object:
......
break;
}
从这个结构,我们可以看到,对象分为zend_iter_object和zend_iter_plain_object, 这是什么意思呢?
复制代码 代码如下:
zend_api enum zend_object_iterator_kind zend_iterator_unwrap(
zval *array_ptr, zend_object_iterator **iter tsrmls_dc)
{
switch (z_type_p(array_ptr)) {
case is_object:
if (z_obj_ht_p(array_ptr) == &iterator_object_handlers) {
*iter = (zend_object_iterator *)zend_object_store_get_object(array_ptr tsrmls_cc);
return zend_iter_object;
}
if (hash_of(array_ptr)) {
return zend_iter_plain_object;
}
return zend_iter_invalid;
case is_array:
if (hash_of(array_ptr)) {
return zend_iter_plain_array;
}
return zend_iter_invalid;
default:
return zend_iter_invalid;
}
}
这就要讲到php的内置接口iterator了,php5开始支持了接口, 并且内置了iterator接口, 所以如果你定义了一个类,并实现了iterator接口,那么你的这个类对象就是zend_iter_object,否则就是zend_iter_plain_object.
对于zend_iter_plain_object的类,foreach会通过hash_of获取该对象的默认属性数组,然后对该数组进行foreach.
而对于zend_iter_object的类对象,则会通过调用对象实现的iterator接口相关函数来进行foreach,iterator接口:
复制代码 代码如下:
iterator extends traversable {
/* 方法 */
abstract public mixed current ( void )
abstract public scalar key ( void )
abstract public void next ( void )
abstract public void rewind ( void )
abstract public boolean valid ( void )
}
所以, 对于这道笔试题, 可以作出如下的答案:
复制代码 代码如下:
class sample implements iterator
{
private $_items = array(1,2,3,4,5,6,7);
public function __construct() {
;//void
}
public function rewind() { reset($this->_items); }
public function current() { return current($this->_items); }
public function key() { return key($this->_items); }
public function next() { return next($this->_items); }
public function valid() { return ( $this->current() !== false ); }
}
$sa = new sample();
foreach($sa as $key => $val){
print $key . "=>" .$val;
}
以上代码在我的php 5.3下运行正常。
上一篇: 流程控制(if、while、for)
下一篇: Oracle中的rownum