查看php文档,我们可以看到Generator实际也是Iterator的具体实现,yield调用时就是返回的Generator对象。
那么怎么理解迭代器和生成器的关系呢?
其实,生成器是迭代器的实现+yield,产生了生成器对象。
我们也可以自己定义一个类似yield的函数,如下:
function myYeild(){ $args = func_get_args(); return new Sample($args); } $generator = myYeild(1,2,3); foreach ($generator as $value) { echo "$value\n"; }
注意,我们的myYeild,是不能和php内置的yeild那么使用的,因为yeild会保存调用上下文,临时离开,并没有return。
这里只是类比一下。
既然yeild可以把普通的对象包装成generator,那么我们的iterator通过yeild也可以像Generator一样吗?
答案有点悲伤,yeild是把传入的值作为参数生成Generator实例,它并不知道我们的iterator。不过这样设计也是合理的,
以防我们自己的iterator不靠谱。
实际使用场合
- 数据库遍历
可以结合游标,遍历数据库时,不需要一次返回所有数据,而是每次取一行。
class AllUser implements \Iterator { protected $index = 0; protected $data = []; public function __construct() { $link = mysqli_connect('192.168.0.91', 'root', '123', 'xxx'); $rec = mysqli_query($link, 'select id from doc_admin'); $this->data = mysqli_fetch_all($rec, MYSQLI_ASSOC); } //1 重置迭代器 public function rewind() { $this->index = 0; } //2 验证迭代器是否有数据 public function valid() { return $this->index < count($this->data); } //3 获取当前内容 public function current() { $id = $this->data[$this->index]; return User::find($id); } //4 移动key到下一个 public function next() { return $this->index++; } //5 迭代器位置key public function key() { return $this->index; } } //实现迭代遍历用户表 $users = new AllUser(); //可实时修改 foreach ($users as $user){ $user->add_time = time(); $user->save(); }
- 文件遍历
一次读取一行 - 实现Iterator接口,让普通类可以使用foreach遍历。
- 协程,参见鸟哥则这篇文章。
注意:可以在生成器的函数前加"&",可以使用引用。在函数里直接return会终止生成器。
更多关于PHP相关内容感兴趣的读者可查看本站专题:《php常用函数与技巧总结》、《php字符串(string)用法总结》、《PHP数组(Array)操作技巧大全》、《PHP数据结构与算法教程》及《php程序设计算法总结》
希望本文所述对大家PHP程序设计有所帮助。