剖析 PHP 中的输出缓冲


Posted in PHP onDecember 21, 2006

我们先来看一段代码:

    〈?php
    for ($i=10; $i〉0; $i--)
    {
     echo $i;
     flush();
     sleep(1);
    }
    ?〉

    按照php手册里的说法,该函数将当前为止程序的所有输出发送到用户的浏览器。

    上面的这段代码,应该隔一秒钟输出一次$i。但是实际中却不一定是这样。有可能是等了10秒钟后,所有的输出同时呈现出来。

    好,我们来改一下这段代码,改成

    〈?php
    ob_end_clean();//修改部分
    for ($i=10; $i〉0; $i--)
    {
     echo $i;
     flush();
     sleep(1);
    }
    ?〉
    嘿,加了这一句ob_end_clean();,居然就OK了。实际上,我们把ob_end_clean()换成ob_end_flush()也一样OK。

    我再来改一改。

    〈?php
    for ($i=10; $i〉0; $i--)
    {
     echo $i;
     ob_flush();//修改部分
     flush();
     sleep(1);
    }
    ?〉
    运行一下,是不是发现$i也隔一秒输出一次了?这是为什么呢?
    别急,我们来看看php.ini。

    打开php.ini,搜索output_buffering,我们会看到类似这样的设置 output_buffering = 4096。正如它的名字output_buffering一样,这个设置的作用就是把输出缓冲一下,缓冲大小为4096bytes.

    在我们的第一段代码里,之所以没有按预期的输出,正是因为这个output_buffering把那些输出都缓冲了。没达到4096bytes或者脚本结束,输出是不会被发送出去的。

    而第二段代码中的ob_end_clean()和ob_end_flush()的作用,就是终止缓冲。这样就不用等到有4096bytes的缓冲之后才被发送出去了。

    第三段代码中,用了一句ob_flush(),它的作用就是把缓冲的数据发送出去,但是并不会终止缓冲,所以它必须在每次flush()前使用。

    如果不想使用ob_end_clean(),ob_end_flush()和ob_flush(),我们就必须把php.ini里的output_buffering设得足够小,例如设为0。需要注意的是,如果你打算在脚本中使用ini_set(”output_buffering”,”0″)来设置,那么请停下来吧,这种方法是不行的。因为在脚本一开始的时候,缓冲设置就已经被载入,然后缓冲就开始了。

    可能你会问了,既然ob_flush()是把缓冲的数据发送出去,那么为什么还需要用flush()???直接用下面这段代码不行吗??

    〈?php
    for ($i=10; $i〉0; $i--)
    {
     echo $i;
     ob_flush();
     sleep(1);
    }
    ?〉
    请注意ob_flush()和flush()的区别。前者是把数据从PHP的缓冲中释放出来,后者是把不在缓冲中的或者说是被释放出来的数据发送到浏览器。所以当缓冲存在的时候,我们必须ob_flush()和flush()同时使用。

    那是不是flush()在这里就是不可缺少的呢?不是的,我们还有另外一种方法,使得当有数据输出的时候,马上被发送到浏览器。下面这两段代码就是不需要使用flush()了。(当你把output_buffering设为0的时候,连ob_flush()和ob_end_clean()都不需要了)

    〈?php
    ob_implicit_flush(true);
    for ($i=10; $i〉0; $i--)
    {
     echo $i;
     ob_flush();
     sleep(1);
    }
    ?〉
    〈?php
    ob_end_clean();
    ob_implicit_flush(true);
    for ($i=10; $i〉0; $i--)
    {
     echo $i;
     sleep(1);
    }
    ?〉
    请注意看上面的ob_implicit_flush(true),这个函数强制每当有输出的时候,即刻把输出发送到浏览器。这样就不需要每次输出(echo)后,都用flush()来发送到浏览器了。

    以上所诉可能在某些浏览器中不成立。因为浏览器也有自己的规则。我是用Firefox1.5,IE6,opera8.5来测试的。其中opera就不能正常输出,因为它有一个规则,就是不遇到一个HTML标签,就绝对不输出,除非到脚本结束。而FireFox和IE还算比较正常的。

    最后附上一段非常有趣的代码,作者为PuTTYshell。在一个脚本周期里,每次输出,都会把前一次的输出覆盖掉。
    以下代码只在firefox下可用,其他浏览器并不支持multipart/x-mixed-replace的Content-Type.

    〈?php
      header('Content-type: multipart/x-mixed-replace;boundary=endofsection');
      print “\n--endofsection\n“;

      $pmt = array(“-“, “\\“, “|“, “/“ );
      for( $i = 0; $i 〈10; $i ++ ){
         sleep(1);
         print “Content-type: text/plain\n\n“;
         print “Part $i\t“.$pmt[$i % 4];
         print “--endofsection\n“;
         ob_flush();
         flush();
      }
      print “Content-type: text/plain\n\n“;
      print “The end\n“;
      print “--endofsection--\n“;
    ?〉 

PHP 相关文章推荐
深入了解php4(2)--重访过去
Oct 09 PHP
ADODB类使用
Nov 25 PHP
PHP set_time_limit(0)长连接的实现分析
Mar 02 PHP
7个超级实用的PHP代码片段
Jul 11 PHP
php一个找二层目录的小东东
Aug 02 PHP
Discuz7.2版的faq.php SQL注入漏洞分析
Aug 06 PHP
PHP中preg_match正则匹配中的/u、/i、/s含义
Apr 17 PHP
thinkphp中字符截取函数msubstr()用法分析
Jan 09 PHP
利用PHP绘图函数实现简单验证码功能的方法
Oct 18 PHP
PHP新特性详解之命名空间、性状与生成器
Jul 18 PHP
IOS 开发之NSDictionary转换成JSON字符串
Aug 14 PHP
PHP删除数组中指定下标的元素方法
Feb 03 PHP
PHP 的几个配置文件函数
Dec 21 #PHP
使用 MySQL 开始 PHP 会话
Dec 21 #PHP
PHP 编程请选择正确的文本编辑软件
Dec 21 #PHP
PHP 模板高级篇总结
Dec 21 #PHP
PHP函数utf8转gb2312编码
Dec 21 #PHP
小偷PHP+Html+缓存
Dec 20 #PHP
PR值查询 | PageRank 查询
Dec 20 #PHP
You might like
PHP遍历目录函数opendir()、readdir()、closedir()、rewinddir()总结
2014/11/18 PHP
PHP实现QQ登录实例代码
2016/01/14 PHP
PHP如何读取由JavaScript设置的Cookie
2017/03/22 PHP
laravel中Redis队列监听中断的分析
2020/09/14 PHP
JQuery Ajax通过Handler访问外部XML数据的代码
2010/06/01 Javascript
js触发select onchange事件的小技巧
2014/08/05 Javascript
JavaScript中的parse()方法使用简介
2015/06/12 Javascript
改变checkbox默认选中状态及取值的实现代码
2016/05/26 Javascript
JavaScript 中 avalon绑定属性总结
2016/10/19 Javascript
EasyUi 打开对话框后控件赋值及赋值后不显示的问题解决办法
2017/01/19 Javascript
Sublime Text新建.vue模板并高亮(图文教程)
2017/10/26 Javascript
Angular4自制一个市县二级联动组件示例
2017/11/21 Javascript
JavaScript 数组去重并统计重复元素出现的次数实例
2017/12/14 Javascript
React Native 自定义下拉刷新上拉加载的列表的示例
2018/03/01 Javascript
vuejs选中当前样式active的实例
2018/08/22 Javascript
原生JS实现轮播图效果
2018/10/12 Javascript
vue路由跳转传参数的方法
2019/05/06 Javascript
亲自动手实现vue日历控件
2019/06/26 Javascript
Vue2.0 实现页面缓存和不缓存的方式
2019/11/12 Javascript
Vue快速实现通用表单验证的方法
2020/02/24 Javascript
vue.js watch经常失效的场景与解决方案
2021/01/07 Vue.js
[01:01:35]Optic vs paiN 2018国际邀请赛小组赛BO2 第二场 8.19
2018/08/21 DOTA
python通过floor函数舍弃小数位的方法
2015/03/17 Python
Python中的数据对象持久化存储模块pickle的使用示例
2016/03/03 Python
利用Python中SocketServer 实现客户端与服务器间非阻塞通信
2016/12/15 Python
和孩子一起学习python之变量命名规则
2018/05/27 Python
BP神经网络原理及Python实现代码
2018/12/18 Python
python3 反射的四种基本方法解析
2019/08/26 Python
使用django和vue进行数据交互的方法步骤
2019/11/11 Python
nginx+uwsgi+django环境搭建的方法步骤
2019/11/25 Python
Python zip函数打包元素实例解析
2019/12/11 Python
详解解决jupyter不能使用pytorch的问题
2021/02/18 Python
五分钟学会HTML5的WebSocket协议
2019/11/22 HTML / CSS
波兰运动鞋网上商店:Distance.pl
2020/07/30 全球购物
质监局领导班子对照检查材料思想汇报
2014/09/27 职场文书
python中24小时制转换为12小时制的方法
2021/06/18 Python