设为首页 收藏本站
查看: 613|回复: 0

[经验分享] 关于PHP的stream_bucket的认识

[复制链接]
发表于 2018-12-14 09:46:43 | 显示全部楼层 |阅读模式
  最近使用stream进行一些项目的开发,在使用到stream bucket时,查阅了google还是国内的百度,发现网上关于stream bucket的介绍文章很少,如php.net的官方手册对于这一块也没有具体的文档,也没有很详细地介绍这个bucket在stream中的作用。最多出现可用信息也就是stream_bucket_append — Append bucket to brigade、stream_bucket_make_writeable — Return a bucket object from the brigade for operating on、stream_bucket_new — Create a new bucket for use on the current stream、stream_bucket_prepend。 相信很多Phper在阅读到这里也会有一些困惑,特别是使用到php_user_filter时必然会接触到bucket。因此我将自己一些理解和应用实例记录下来,同时我也希望跟各位能有进一步的沟通,可以email我,lailaiji@163.com(赖来基)。
  在PHP中,可以将文件、网络、数据压缩等操作资源对象可以视为一个stream流对象,通过统一流操作实现对这些对象的抽象和封装,方便我们处理相关资源对象。这篇博文就没有涉及展开对流的细节讲解,而是侧重通过讲解php_user_filter带来对于stream bucket的一些认识。
  回到本文的正题,一开始我们需要先了解清楚两个概念,Bucket(桶)及Brigade(队列)。这里可以理解为装载数据的一个对象,该对象包含数据内容(data)及数据长度(datalen),相当于我们把流数据装到这个桶里来,该桶除了有数据还有数据的高度;而Brigade呢,则是这些桶的一个队列,从而组织成为一个数据流,这个队列可以是一个或者多个桶组成。
  理解了上面的概念,我们通过自定义的一个filter来分步讲解这两个概念。
  新建一个test_custom_filter.php
class foo extends php_user_filter {
  public function filter($in, $out, &$consumed, $closing) {
  $bucket = stream_bucket_make_writeable($in);
  if($bucket)
        {
            $consumed += $bucket->datalen;
            $_f = fopen("php://temp",'w');
            $new_bucket = stream_bucket_new($_f, '');
            $string     = $bucket->data."I am lailaiji\n";
            $new_bucket->data=ucfirst($string);
            fclose($_f);
            stream_bucket_append($out, $new_bucket);
        }
        //数据处理成功,可供其它管道读取
        return PSFS_PASS_ON;
  }
}
stream_filter_register('test', 'foo');
$data = file_get_contents('php://filter/read=test/resource=data.txt');
echo $data;  而data.txt文件
hello  运行php -f test_custom_filter.php得到如下
Hello
I am lailaiji  自定义的filter需要继承基类php_user_filter,并且重写filter方法,这个方法共有四个参数

  •   $in: 指向输入流队列
  •   $out:指向输出流队列
  •   $consumed: 引用方式传递,必须增加了转换数据的长度
  •   $closing: 如果最后一个bucket或者即将关闭时,可以设置为true
  

  认识第一个函数:stream_bucket_make_writeable($bregade)
  用途:将队列里的每个bucket取出,得到一个bucket对象。
$bucket = stream_bucket_make_writeable($in);  该方法可以得到我们从data.txt文件中的内容,可以通过调用bucket对象属性$bucket->data
  第二个函数:stream_bucket_new($handle,$buffer)
  用途:产生一个指定流的bucket对象,一般来说,我们都会使用临时php://temp进行操作流数据
  第三个函数:stream_bucket_append($handle,$bucket) 或者stream_bucket_prepend($handle,$bucket)
  用途,将bucket对象追加到队列里
stream_bucket_append($out, $new_bucket);  上面通过这个方法将我们的bucket追加到输出流($out),因此我们可以通过file_get_contents获得输出内容
  

  另外php_user_filter还定义了三个常量
  PSFS_PASS_ON:过滤器处理成功,输出流有可用数据返回
  PSFS_FEED_ME:过滤器处理成功,但是没有数据返回
  PSFS_ERR_FATAL:默认返回值,有错误,不返回
  

  还有一点需要说明的是:
  如果仔细的读者在使用php_user_filter,会发现filter函数必然后运行两次或者两次以上,明明我只有一个输入流,怎么会被运行两次呢?后来仔细分析了才发现最后一次运行是由于最后一个bucket即将关闭时,也就是closing的值为true的时候被调起,为整理关闭前做一些处理,此时输入流输出为空




运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-651196-1-1.html 上篇帖子: php配置时,报错apxs没安装 下篇帖子: PHP5.3安装Zend Guard Loader图文教程
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表