SPL提供了6个迭代器接口:
| Traversable |
遍历接口(检测一个类是否可以使用 foreach 进行遍历的接口) |
| Iterator |
迭代器接口(可在内部迭代自己的外部迭代器或类的接口) |
| IteratorAggregate |
聚合式迭代器接口(创建外部迭代器的接口) |
| OuterIterator |
迭代器嵌套接口(将一个或多个迭代器包裹在另一个迭代器中) |
| RecursiveIterator |
递归迭代访问接口(提供递归访问功能) |
| SeekableIterator |
可索引迭代访问接口(实现查找功能) |
下面对各种迭代器接口简单介绍一下:
1. Traversable
Traversable接口实际上不是一个接口,在实际写php代码中不能用。因为只有内部的PHP类(用C写的类)才可以直接实现Traversable接口。可以说这是个特性级别的东西。实际的PHP编程中我们使用Iterator接口或者IteratorAggregate接口来实现遍历。
Traversable 接口不能直接实现(implements).Traversable 重要的一个用处就是判断一个类是否可以遍历:
1 |
if($class instanceof Traversable) |
下面是官方例子:
2 |
if( !is_array( $items ) && !$items instanceof Traversable ) |
2. Iterator
Iterator接口的主要用途是允许一个类实现一个基本的迭代功能,从而使它可以被循环访问,根据键值访问以及回滚。Iterator接口摘要如下:
01 |
Iterator extends Traversable |
04 |
abstract public mixed current(void) |
06 |
abstract public scalar key(void) |
08 |
abstract public void next(void) |
10 |
abstract public void rewind(void) |
12 |
abstract public boolean valid(void) |
外部迭代器接口,实现该接口的对象可以迭代自己内部的数据。
Iterator 的例子这里就不再列举了,本专题前面部分以后后续有很多例子,具体请自行查看。
3. IteratorAggregate
又叫聚合式迭代器。创建外部迭代器的接口,其摘要如下:
1 |
IteratorAggregate extends Traversable { |
3 |
abstract public Traversable getIterator ( void ) |
其中getIterator 方法返回值必须是能遍历或实现Iterator接口(must be traversable or implement interface Iterator)。SPL还提供了一些专门用来与IteratorAggregate接口一起使用的内置迭代器。使用这些迭代器意味着只需要实现一个方法并实例化一个类就可以使对象可以迭代访问了。
实例:
04 |
class myData implements IteratorAggregate |
06 |
public $property1 = "公共属性1"; |
07 |
public $property2 = "公共属性2"; |
08 |
public $property3 = "公共属性3"; |
10 |
public function __construct() |
12 |
$this->property4 = "最后一个公共属性"; |
15 |
public function getIterator() |
17 |
return new ArrayIterator($this); |
22 |
foreach ($obj as $key => $value) { |
23 |
echo "键名:{$key} 值:{$value}
"; |
程序输出:
4 |
键名:property4 值:最后一个公共属性 |
4. ArrayAccess
数组式访问接口。实现该接口的对象能像数组一样使用:
3 |
abstract public boolean offsetExists ( mixed $offset ) |
4 |
abstract public mixed offsetGet ( mixed $offset ) |
5 |
abstract public void offsetSet ( mixed $offset , mixed $value ) |
6 |
abstract public void offsetUnset ( mixed $offset ) |
- ArrayAccess::offsetExists — 检查一个偏移位置是否存在
- ArrayAccess::offsetGet — 获取一个偏移位置的值
- ArrayAccess::offsetSet — 设置一个偏移位置的值
- ArrayAccess::offsetUnset — 复位一个偏移位置的值
举个栗子:
04 |
class obj implements arrayaccess { |
05 |
private $container = array(); |
06 |
public function __construct() { |
07 |
$this->container = array( |
13 |
public function offsetSet($offset, $value) { |
14 |
if (is_null($offset)) { |
15 |
$this->container[] = $value; |
17 |
$this->container[$offset] = $value; |
20 |
public function offsetExists($offset) { |
21 |
return isset($this->container[$offset]); |
23 |
public function offsetUnset($offset) { |
24 |
unset($this->container[$offset]); |
26 |
public function offsetGet($offset) { |
27 |
return isset($this->container[$offset]) ? $this->container[$offset] : null; |
33 |
var_dump(isset($obj["two"])); |
34 |
var_dump($obj["two"]); |
36 |
var_dump(isset($obj["two"])); |
37 |
$obj["two"] = "A value"; |
38 |
var_dump($obj["two"]); |
5. Serializable
序列化接口。实现该接口的类不能使用__sleep() 和__wakeup().在serialize时不执行__destruct(),在unserialize不执行__construct()。
3 |
abstract public string serialize ( void ) |
4 |
abstract public mixed unserialize ( string $serialized ) |
实现此接口的类将不再支持 __sleep() 和 __wakeup()。不论何时,只要有实例需要被序列化,serialize 方法都将被调用。它将不会调用 __destruct() 或有其他影响,除非程序化地调用此方法。当数据被反序列化时,类将被感知并且调用合适的 unserialize() 方法而不是调用 __construct()。如果需要执行标准的构造器,你应该在这个方法中进行处理。
- Serializable::serialize — 对象的字符串表示
- Serializable::unserialize — 构造对象
3 |
abstract public string serialize ( void ) |
4 |
abstract public mixed unserialize ( string $serialized ) |
例子:
01 |
class obj implements Serializable { |
03 |
public function __construct() { |
04 |
$this->data = "My private data"; |
06 |
public function serialize() { |
07 |
return serialize($this->data); |
09 |
public function unserialize($data) { |
10 |
$this->data = unserialize($data); |
12 |
public function getData() { |
18 |
$ser = serialize($obj); |
20 |
$newobj = unserialize($ser); |
22 |
var_dump($newobj->getData()); |
6. Closure
4 |
public static Closure bind ( Closure $closure , object $newthis [, mixed $newscope = 'static' ] ) |
5 |
public Closure bindTo ( object $newthis [, mixed $newscope = 'static'] ) |
这个具体还没研究,具体可以去看官方文档:http://www.php.net/manual/en/reserved.interfaces.php