剖析PHP中的输出缓冲

转载 2012年03月24日 22:23:18

剖析PHP中的输出缓冲

本文按署名·非商业用途·保持一致授权
作者:  ,发表于2005年12月24日01时54分 

我们先来看一段代码。

<?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";
?>

http://www.laruence.com/2010/04/15/1414.html

15 Apr 10 深入理解ob_flush和flush的区别 
<iframe allowtransparency="true" frameborder="0" hspace="0" id="I1_1332598580773" marginheight="0" marginwidth="0" name="I1_1332598580773" scrolling="no" src="https://plusone.google.com/_/+1/fastbutton?url=http%3A%2F%2Fwww.laruence.com%2F2010%2F04%2F15%2F1414.html&amp;size=small&amp;count=true&amp;hl=en-US&amp;jsh=m%3B%2F_%2Fapps-static%2F_%2Fjs%2Fgapi%2F__features__%2Frt%3Dj%2Fver%3DLJdoFpm7lzE.zh_CN.%2Fsv%3D1%2Fam%3D!brN6X75-Zu-IDRYPeA%2Fd%3D1#id=I1_1332598580773&amp;parent=http%3A%2F%2Fwww.laruence.com&amp;rpctoken=289697630&amp;_methods=onPlusOne%2C_ready%2C_close%2C_open%2C_resizeMe%2C_renderstart" tabindex="0" vspace="0" width="100%" title="+1" style="margin-top: 0px; margin-right: 0px; margin-bottom: 0px; margin-left: 0px; padding-top: 0px; padding-right: 0px; padding-bottom: 0px; padding-left: 0px; border-top-width: 0px; border-right-width: 0px; border-bottom-width: 0px; border-left-width: 0px; border-style: initial; border-color: initial; font-weight: inherit; font-style: inherit; font-family: inherit; vertical-align: baseline; width: 70px; border-top-style: none; border-right-style: none; border-bottom-style: none; border-left-style: none; height: 15px; position: static; left: 0px; top: 0px; visibility: visible; "></iframe>

ob_flush/flush在手册中的描述, 都是刷新输出缓冲区, 并且还需要配套使用, 所以会导致很多人迷惑…

其实, 他们俩的操作对象不同, 有些情况下, flush根本不做什么事情..

ob_*系列函数, 是操作PHP本身的输出缓冲区.

所以, ob_flush是刷新PHP自身的缓冲区.

而flush, 严格来讲, 这个只有在PHP做为apache的Module(handler或者filter)安装的时候, 才有实际作用. 它是刷新WebServer(可以认为特指apache)的缓冲区.

在apache module的sapi下, flush会通过调用sapi_module的flush成员函数指针, 间接的调用apache的api: ap_rflush刷新apache的输出缓冲区, 当然手册中也说了, 有一些apache的其他模块, 可能会改变这个动作的结果..

  1. 有些Apache的模块,比如mod_gzip,可能自己进行输出缓存,
  2. 这将导致flush()函数产生的结果不会立即被发送到客户端浏览器。
  3.  
  4. 甚至浏览器也会在显示之前,缓存接收到的内容。例如 Netscape
  5. 浏览器会在接受到换行或 html 标记的开头之前缓存内容,并且在
  6. 接受到 </table> 标记之前,不会显示出整个表格。
  7.  
  8. 一些版本的 Microsoft Internet Explorer 只有当接受到的256个
  9. 字节以后才开始显示该页面,所以必须发送一些额外的空格来让这
  10. 些浏览器显示页面内容。

所以, 正确使用俩者的顺序是. 先ob_flush, 然后flush,

当然, 在其他sapi下, 不调用flush也可以, 只不过为了保证你代码的可移植性, 建议配套使用.



相关文章推荐

PHP逐行输出数据并解决两种常见缓冲问题

PHP逐行输出数据并解决两种常见缓冲问题

PHP输出缓冲控制(Output Control)总结

php 缓冲简介 其实我对php ob 系列印象还是很模糊,具体怎么玩的,还不是很了解,平时curd,确实对这些内容没有深入。作为phper 甚是惭愧。网上搜了一通,互相copy,代码运行不能出...

PHP中刷新输出缓冲详解

buffer是一个内存地址空间,Linux系统默认大小一般为4096(1kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的设备之间传办理数据的区域。通过buffer,可以使进程这间的相...
  • leconcq
  • leconcq
  • 2011年07月23日 17:42
  • 3314

PHP输出缓冲控制 - Output Control 函数 应用详解

转载出自链接:http://my.oschina.net/whrlmc/blog/85782  简介 说到输出缓冲,首先要说的是一个叫做缓冲器(buffer)的东西。举个简单的例子说明他的作...
  • adslcom
  • adslcom
  • 2012年12月11日 14:37
  • 524

PHP刷新输出缓冲

buffer 是一个内存地址空间,Linux系统默认大小一般为4096(1kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的 设备之间传办理数据的区域。通过buffer,可以使进程...

PHP中刷新输出缓冲

PHP中刷新输出缓冲 buffer是一个内存地址空间,Linux系统默认大小一般为4096(1kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的 设备之间传办理数据的区域。通过buf...

再说php输出缓冲控制那些事

之前转了一篇关于php输出缓冲控制的文章:http://blog.csdn.net/adslcom/article/details/8282342 用来方便之后查看的。对于内容的例子代码也没有一一测试...
  • adslcom
  • adslcom
  • 2013年01月06日 22:11
  • 768

我所理解的php缓冲机制及嵌套级别

我理解的要点: 1、所有缓冲区控制是在一个PHP执行进程中发生的。如:你打开n个demo.php,他们之间开启和关闭缓冲是互不影响的。 2、output_buffering在程序中用ini_...
  • soonfly
  • soonfly
  • 2016年08月03日 17:19
  • 1206

转开源中国的php ob_start()、ob_end_flush和ob_end_clean()多级缓冲

ob_start() 和 ob_end_flush() 是一对很好的搭档,可以实现对输出的控制。当成一对出现理解起来就没什么问题,但是当他们两个各自出现次数增加时,就比较难理解了....
  • soonfly
  • soonfly
  • 2016年08月03日 15:07
  • 497

php缓冲 output_buffering和ob_start

http://www.perfgeeks.com/?p=344php缓冲 output_buffering和ob_startbufferbuffer是一个内存地址空间,Linux系统默认大小一般为40...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:剖析PHP中的输出缓冲
举报原因:
原因补充:

(最多只允许输入30个字)