2009年12月2日星期三

剖析 PHP 中的输出缓冲

FROM: 已不知出去

[php]
for ($i=10; $i > 0; $i--) {
echo $i;
flush();
sleep(1);
}
[/php]

按照 PHP 手册里的说法, flash() 该函数将当前为止程序的所有输出发送到用户的浏览器。所以,上面的这段代码,应该隔一秒钟输出一次 $i。但是实际中却不一定是这样,有可能是等 10 秒钟后,所有的输出才同时呈现出来。

改一下这段代码:
[php]
ob_end_clean();//修改部分
for ($i = 10; $i > 0; $i--) {
echo $i;
flush();
sleep(1);
}
[/php]

加了 ob_end_clean();,可以了。实际上,把 ob_end_clean() 换成 ob_end_flush() 也一样行。

再改成如下:
[php]
for ($i=10; $i > 0; $i--) {
echo $i;
ob_flush();//修改部分
flush();
sleep(1);
}
[/php]

运行一下,发现 $i 也是隔一秒输出一次。这是为什么呢?看 PHP 的配置文件 php.ini。
打开 php.ini,搜索 output_buffering,会看到类似这样的设置 output_buffering = 4096。这个设置的作用就是把输出缓冲,缓冲大小为 4096 bytes.

在第一段代码里,之所以没有按预期的输出,正是因为这个 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);
}
[/php]

这里需清楚 ob_flush()flush()的区别。前者是把数据从 php 的缓冲中释放出来,后者是把不在缓冲中的或者说是被释放出来的数据发送到浏览器。所以当缓冲存在的时候,必须 ob_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]

[php]
ob_end_clean();
ob_implicit_flush(true);
for ($i=10; $i > 0; $i--) {
echo $i;
sleep(1);
}
[/php]

请注意看上面的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]

2 条评论:

  1. 为什么上面的代码在firefox中可以,在IE8中就不行呢?(IE7未测)
    IE8中是一下子全输出来。

    回复删除

  2. phper :
    为什么上面的代码在firefox中可以,在IE8中就不行呢?(IE7未测)
    IE8中是一下子全输出来。

    正如文中所说:「以上所诉可能在某些浏览器中不成立。因为浏览器也有自己的规则。我是用firefox1.5,ie6,opera8.5来测试的。其中opera就不能正常 输出,因为它有一个规则,就是不遇到一个html标签,就绝对不输出,除非到脚本结束。而firefox和ie还算比较正常的。」
    按你所说的情况,应是IE8修改了游戏规则,即不遇到一个html标签,就绝对不输出,除非脚本结束了~

    回复删除