PHP迭代器和生成器用法实例分析


Posted in PHP onSeptember 28, 2019

本文实例讲述了PHP迭代器和生成器用法。分享给大家供大家参考,具体如下:

迭代器

迭代器实际是一个实现了Iterator的类,可以用foreach进行遍历。

例如:

<?php
class Sample implements Iterator{
  private $curIndex=0;
  private $items=null;
  public function __construct($_items) {
    $this->items = $_items;
  }
  public function current (){
    echo "current\n";
    return $this->items[$this->curIndex];
  }
  public function key (){
    echo "key\n";
    return $this->curIndex;
  }
  public function next (){   
      echo "next\n"; 
      $this->curIndex++;
  }
  public function rewind (){
      $this->curIndex = 0;   
  }
  public function send ( $value ){
    if($value == "stop"){
      $this->curIndex = null;
    }
  }
  public function valid (){
    echo "valid\n";
    return isset($this->items[$this->curIndex]);
  }
}
$sample = new Sample([1,2,3]);
foreach ($sample as $k =>$v){
}

输出

 valid current key next

可以看到foreach 是先调用valid判断迭代器是否有效,然后再调用current获取当前值,同时调用next移动key到指向下一个值(输出key是因为 $k=>$v的缘故)。

生成器

让我们先看一下官方文档

生成器提供了一种更容易的方法来实现简单的对象迭代,相比较定义类实现 Iterator 接口的方式,性能开销和复杂性大大降低。
生成器允许你在 foreach 代码块中写代码来迭代一组数据而不需要在内存中创建一个数组, 那会使你的内存达到上限,或者会占据可观的处理时间。
相反,你可以写一个生成器函数,就像一个普通的自定义函数一样, 和普通函数只返回一次不同的是, 生成器可以根据需要 yield 多次,以便生成需要迭代的值。
PHP 将会在每次需要值的时候调用生成器函数,并在产生一个值之后保存生成器的状态,这样它就可以在需要产生下一个值的时候恢复调用状态。

下面是php官方文档中的示例

<?php
function gen_one_to_three() {
  for ($i = 1; $i <= 3; $i++) {
    //注意变量$i的值在不同的yield之间是保持传递的。
    yield $i;
  }
}
$generator = gen_one_to_three();
foreach ($generator as $value) {
  echo "$value\n";
}
var_dump($generator); //实际上是Generator对象

如上,若把3修改成10000,对于$generator实际上没有区别,它只是保存了一个当前值(当然还有相关的内部状态,这里是为了简化),并没有产生10000个数。

从中可以看出生成器的优势在于减少内存的使用,在需要时才生成对应的值。

查看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 出现乱码和Sessions验证问题的解决方法!
Dec 06 PHP
php下保存远程图片到本地的办法
Aug 08 PHP
无法在发生错误时创建会话,请检查 PHP 或网站服务器日志,并正确配置 PHP 安装(win+linux)
May 05 PHP
PHP中去掉字符串首尾空格的方法
May 19 PHP
php数据库配置文件一般做法分享
Jul 07 PHP
解析zend studio中直接导入svn中的项目的方法步骤
Jun 21 PHP
php遍历目录输出目录及其下的所有文件示例
Jan 27 PHP
支持中文和其他编码的php截取字符串函数分享(截取中文字符串)
Mar 13 PHP
laravel容器延迟加载以及auth扩展详解
Mar 02 PHP
Swoole-1.7.22 版本已发布,修复PHP7相关问题
Dec 31 PHP
php mysql like 实现多关键词搜索的方法
Oct 29 PHP
PHP实现随机发扑克牌
Apr 22 PHP
php实现的数组转xml案例分析
Sep 28 #PHP
PHP反射原理与用法深入分析
Sep 28 #PHP
Windows服务器中PHP如何安装redis扩展
Sep 27 #PHP
php-fpm超时时间设置request_terminate_timeout资源问题分析
Sep 27 #PHP
thinkPHP+LayUI 流加载实现功能
Sep 27 #PHP
PHP的cookie与session原理及用法详解
Sep 27 #PHP
PHP下载文件函数与用法示例
Sep 27 #PHP
You might like
解析php中memcache的应用
2013/06/18 PHP
PHP制作登录异常ip检测功能的实例代码
2016/11/16 PHP
laravel 使用auth编写登录的方法
2019/09/30 PHP
JavaScript 字符串与数组转换函数[不用split与join]
2009/12/13 Javascript
javascript计时器详解
2015/02/28 Javascript
两款JS脚本判断手机浏览器类型跳转WAP手机网站
2015/10/16 Javascript
AngularJS手动表单验证
2016/02/01 Javascript
Nodejs 搭建简单的Web服务器详解及实例
2016/11/30 NodeJs
jquery仿ps颜色拾取功能
2017/03/08 Javascript
node.js中debug模块的简单介绍与使用
2017/04/25 Javascript
详解Node.js利用node-git-server快速搭建git服务器
2017/09/27 Javascript
浅谈React组件之性能优化
2018/03/02 Javascript
Vue项目报错:Uncaught SyntaxError: Unexpected token
2018/11/10 Javascript
如何制作一个Node命令行图像识别工具
2018/12/12 Javascript
微信小程序实现swiper切换卡内嵌滚动条不显示的方法示例
2018/12/20 Javascript
vue router动态路由设置参数可选问题
2019/08/21 Javascript
vue二选一tab栏切换新做法实现
2021/01/19 Vue.js
[04:36]DOTA2国际邀请赛 ti3精彩集锦
2013/08/19 DOTA
python中查看变量内存地址的方法
2015/05/05 Python
Python代码块批量添加Tab缩进的方法
2018/06/25 Python
flask中过滤器的使用详解
2018/08/01 Python
python pptx复制指定页的ppt教程
2020/02/14 Python
Python编程快速上手——疯狂填词程序实现方法分析
2020/02/29 Python
python 输入字符串生成所有有效的IP地址(LeetCode 93号题)
2020/10/15 Python
REISS英国官网:伦敦High Street最受欢迎品牌
2016/12/21 全球购物
Currentbody西班牙:美容仪专家
2019/09/28 全球购物
市级文明单位申报材料
2014/05/07 职场文书
校园活动策划方案
2014/06/13 职场文书
十一国庆节“向国旗敬礼”主题班会活动方案
2014/09/27 职场文书
老公婚前保证书
2015/02/28 职场文书
2015教师个人德育工作总结
2015/07/22 职场文书
严以律己专题学习研讨会发言材料
2015/11/09 职场文书
早安问候语大全
2015/11/10 职场文书
ObjectMapper 如何忽略字段大小写
2021/06/29 Java/Android
Python实现Matplotlib,Seaborn动态数据图
2022/05/06 Python
MySQL数据库如何查看表占用空间大小
2022/06/10 MySQL