php的flush和ob_flush无效问题解决办法

php 缓冲简介

其实我对php ob
系列印象还是很模糊,具体怎么玩的,还不是很了解,平时curd,确实对这些内容没有深入。作为phper
甚是惭愧。网上搜了一通,互相copy,代码运行不能出现作者所描述现象,本文良心出品,代码都是作者运行过。

当执行输出的时候,比如 echo,print。输出并没有立即送给 web server,
而是将数据写入 php buffer。php output_buffering
机制好处当然提升性能。其实 php 文件最终在浏览器上显示,走过3个缓冲阶段:
php buffer=》web server buffer=》browser buffer。 最后显示到浏览器

默认情况下,php buffer 是开启的,而且该 buffer 默认值是4096,即4
kb。你可以通过在php.ini配置文件中找到output_buffering配置。buffer是一个内存地址空间,Linux系统默认大小一般为4096(4kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的设备之间传办理数据的区域。通过buffer,可以使进程这间的相互等待变少。这里说一个通俗一点的例子,你打开文本编辑器编辑一个文件的时候,你每输入一个字符,操作系统并不会立即把这个字符直接写入到磁盘,而是先写入到buffer,当写满了一个buffer的时候,才会把buffer中的数据写入磁盘,当然当调用内核函数flush()的时候,强制要求把buffer中的脏数据写回磁盘。

buffer
buffer是一个内存地址空间,Linux系统默认大小一般为4096(4kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的设备之间传办理数据的区域。通过buffer,可以使进程这间的相互等待变少。这里说一个通俗一点的例子,你打开文本编辑器编辑一个文件的时候,你每输入一个字符,操作系统并不会立即把这个字符直接写入到磁盘,而是先写入到buffer,当写满了一个buffer的时候,才会把buffer中的数据写入磁盘,当然当调用内核函数flush()的时候,强制要求把buffer中的脏数据写回磁盘。

我的基础环境是nginx1.6.2+php5.3
做一个逐行输出,使用ob_flush时试了N种方法不起作用,比如下面的代码:

举个例子

<?php
echo "南无阿弥陀佛<br>";
header("content-type:text/html;charset='utf-8'");
echo "真善忍好!";
//output
//南无阿弥陀佛
//真善忍好

header()必须在任何实际输出之前调用,但是我们程序已经输出了,却正常运行。在看下面的代码:

<?php

echo "南无阿弥陀佛<br>";
ob_flush();
header("content-type:text/html;charset='utf-8'");
echo "真善忍好!";
//output
//南无阿弥陀佛
//Cannot modify header information - headers already sent by (output started at E:phptest.php:3)
//真善忍好

上面程序说明程序并没有立即输出,而当调用ob_flush
函数的时候才刷新缓冲,输出。

同样的道理,当执行echo,print的时候,输出并没有立即通过tcp传给客户端浏览器显示,
而是将数据写入php buffer。php output_buffering机制,意味在tcp
buffer之前,建立了一新的队列,数据必须经过该队列。当一个php
buffer写满的时候,脚本进程会将php
buffer中的输出数据交给系统内核交由tcp传给浏览器显示。所以,数据会依次写到这几个地方:echo/print -> php buffer -> tcp buffer
-> browser

<?php
ob_start();
for(;;)
{
    echo “<br>…….”;
    ob_flush();
    flush();
    sleep(1);
}
?>

ob_flush() 与 flush()

ob_flush() , flush() 函数php
手册上都有详细的说明,你可以去查阅一下。二者的区别是:

ob_flush() 是刷新PHP自身的缓冲区

flush()是 它是刷新WebServer
服务器的缓冲。输出到浏览器。但是会出现下面的情况:

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

比如:

<?php
/**
Server:LightTPD/1.4.28 (Win32)
X-Powered-By:PHP/5.3.27
*/
echo '佛法无边'."<BR>";
ob_flush();
flush();
sleep(1);
echo '法轮常转';

//output

上面的代码 在 chrome 浏览器上面
是一行一行的输出,在ie系列的浏览器则是全部输出。其实就是上面的
第四条一些浏览器只有当接收256个字符才开始显示。把上面的代码改成下面形式:

<?php
/**
Server:LightTPD/1.4.28 (Win32)
X-Powered-By:PHP/5.3.27
*/
echo str_pad('',240)."n"; 
echo '佛法无边'."<BR>";
ob_flush();
flush();
sleep(1);
echo '法轮常转';

//output

这样在ie下面就会一行一行输出,因为超过256个字符。

php output_buffering
默认情况下,php
buffer是开启的,而且该buffer默认值是4096,即4kb。你可以通过在php.ini配置文件中找到output_buffering配置.当echo,print等输出用户数据的时候,输出数据都会写入到php
output_buffering中,直到output_澳门新浦京娱乐游戏,buffering写满,会将这些数据通过tcp传送给浏览器显示。你也可以通过ob_start()手动激活php
output_buffering机制,使得即便输出超过了4kb数据,也不真的把数据交给tcp传给浏览器,因为ob_start()将php
buffer空间设置到了足够大。只有直到脚本结束,或者调用ob_end_flush函数,才会把数据发送给客户端浏览器。

谷歌了不少的写法都不行,所以问题应该出在了环境配置上而不是使用方法上。
话说还是stackoverflow给力,搜索“php flush not
working”找到了一个正确的解决方法:
检查nginx配置文件(nginx.conf),禁用nginx的buffering:

ob 其他函数说明

1.ob_end_flush 与 ob_end_clean**

end
的顾名思义就结束,关闭缓冲区,都是关闭输出缓冲,一个是输出缓冲区,一个是清除。比如

<?php
/**
Server:LightTPD/1.4.28 (Win32)
X-Powered-By:PHP/5.3.27
*/
echo 'before';
ob_end_clean();
echo str_pad('',4096)."n";
for ($i=10; $i>0; $i--) 
{ 
    echo $i;
    sleep(1);
}

上述代码是一下输出全部内容,而不是一个一个输出。ob_end_clean()
不是关闭了缓冲了?怎么不是一个一个输出呢,其实我们上面也说了,php
不是直接输出给浏览器,而是 web server。 虽然php 没有了 缓冲。但是web
server 还是有的。所以下面代码:

/**
Server:LightTPD/1.4.28 (Win32)
X-Powered-By:PHP/5.3.27
*/
echo 'before';
ob_end_clean();
echo str_pad('',4096)."n";
for ($i=10; $i>0; $i--) 
{ 
    flush();
    echo $i;
    sleep(1);
}

加上flush(),就会一行一行输出。 如果把ob_end_clean 换成 ob_end_flush 
会把 before 输出来。

其他函数 可参考手册,比较简单。

1.当output_buffering=4096,并且输出较少数据(少于一个buffer) 复制代码 代码如下:
for ($i = 0; $i < 10; $i++) {
echo $i . ‘
‘;
sleep($i + 1); //
}
?>

proxy_buffering off;
gzip off;
fastcgi_keep_conn on;

总结

php 脚本到浏览器,要经过  php buffer=》web server buffer=》browser
buffer。 最后显示到浏览器。 缺一不可。 所以我们要 ob_flush 和 
flush 以及加上  echo str_pad(”,4096) 才能调试出你想要的效果。

现象:不是每隔几秒就会有间断性输出,而是直到响应结束,才能看一次性看到输出,在等待服务器脚本处理结束之前,浏览器界面一直保持空白。这是因为,数据量太小,php
output_buffering没有写满。写数据的顺序,依次是echo->php
buffer->tcp buffer->browser

要注意最后这句fastcgi的哦~~
检查php.ini,禁用buffering:

2.当output_buffering=0,并且输出较少数据(少于一个buffer) 复制代码 代码如下:
//通过ini_set(‘output_buffering’, 0)并不生效
//应该编辑/etc/php.ini,设置output_buffering=0禁用output
buffering机制
//ini_set(‘output_buffering’, 0); //彻底禁用output buffering功能
for ($i = 0; $i < 10; $i++) {
echo $i . ‘
‘;
flush(); //通知操作系统底层,尽快把数据给客户端浏览器
sleep($i + 1); //
}
?>

output_buffering = off

现象:与刚才显示并不一致,禁用了php
buffering机制之后,在浏览器可以断断续续看到间断性输出,不必等到脚本执行完毕才看到输出。这是因为,数据没有在php
output_buffering中停留。写数据的顺序依次是echo->tcp
buffer->browser

注意这句配置不能通过ini_set()函数动态在程序中设置,这在php官方手册中有说明:
the output_buffering setting is PHP_INI_PERDIR therefore it may not
be set using ini_set()
经过上面两步的配置(nginx.conf和php.ini)后,重启nginx就可以了,再次测试文章开头的代码,成功逐行输出

3.当output_buffering=4096.,输出数据大于一个buffer,不调用ob_start() 复制代码 代码如下:
#//创建一个4kb大小的文件
$dd if=/dev/zero of=f4096 bs=4096 count=1
for ($i = 0; $i < 10; $i++) {
echo file_get_contents(‘./f4096’) . $i . ‘
‘;
sleep($i +1);
}
?>

补充:PHP flush()与ob_flush()的区别

现象:响应还没结束(http连接没有关闭),断断续续可以看到间断性输出,浏览器界面不会一直保持空白。尽管启用了php
output_buffering机制,但依然会间断性输出,而不是一次性输出,是因为output_buffering空间不够用。每写满一个php
buffering,数据就会发送到客户端浏览器。

buffer —- flush()
 
buffer是一个内存地址空间,Linux系统默认大小一般为4096(1kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的
设备之间传办理数据的区域。通过buffer,可以使进程这间的相互等待变少。这里说一个通俗一点的例子,你打开文本编辑器编辑一个文件的时候,你每输入
一个字符,操作系统并不会立即把这个字符直接写入到磁盘,而是先写入到buffer,当写满了一个buffer的时候,才会把buffer中的数据写入磁
盘,当然当调用内核函数flush()的时候,强制要求把buffer中的脏数据写回磁盘。
同样的道理,当执行echo,print的时候,输出并没有立即通过tcp传给客户端浏览器显示,
而是将数据写入php buffer。php output_buffering机制,意味在tcp
buffer之前,建立了一新的队列,数据必须经过该队列。当一个php
buffer写满的时候,脚本进程会将php
buffer中的输出数据交给系统内核交由tcp传给浏览器显示。所以,数据会依次写到这几个地方echo/pring
-> php buffer -> tcp buffer -> browser

4.当output_buffering=4096, 输出数据大于一个tcp buffer,
调用ob_start()
复制代码 代码如下:
ob_start(); //开启php buffer
for ($i = 0; $i < 10; $i++) {
echo file_get_contents(‘./f4096’) . $i . ‘
‘;
sleep($i + 1);
}
ob_end_flush();
?>

php output_buffering — ob_flush()

现象:直到服务端脚本处理完成,响应结束,才看到完整输,输出间隔时间很短,以至你感受不到停顿。在输出之前,浏览器一直保持着空白界面,等待服务端数据。这是因为,php一旦调用了ob_start()函数,它会将php
buffer扩展到足够大,直到ob_end_flush函数调用或者脚本运行结速才发送php
buffer中的数据到客户端浏览器。

默认情况下,php
buffer是开启的,而且该buffer默认值是4096,即1kb。你可以通过在php.ini配置文件中找到output_buffering配置.当echo,print等输出用户数据的时候,输出数据都会写入到php
output_buffering中,直到output_buffering写满,会将这些数据通过tcp传送给浏览器显示。你也可以通过
ob_start()手动激活php
output_buffering机制,使得即便输出超过了1kb数据,也不真的把数据交给tcp传给浏览器,因为ob_start()将php
buffer空间设置到了足够大
。只有直到脚本结束,或者调用ob_end_flush函数,才会把数据发送给客户端浏览器。

output buffering函数

这两个函数的使用怕是很多人最迷惑的一个问题,手册上对两个函数的解释也语焉不详,没有明确的指出它们的区别,似乎二者的功能都是刷新输出缓存。但在我们文章一开始的代码中如果讲fush()替换成ob_flush(),程序就再不能正确执行了。显然,它们是有区别的,否则也手册中直接说明其中一个是另外一个函数的别名即可了,没必要分别说明。那么它们的区别到底是什么呢?

1.ob_get_level**
返回输出缓冲机制的嵌套级别,可以防止模板重复嵌套自己。

在没有开启缓存时,脚本输出的内容都在服务器端处于等待输出的状态
,flush()可以将等待输出的内容立即发送到客户端。

1.ob_start
激活output_buffering机制。一旦激活,脚本输出不再直接出给浏览器,而是先暂时写入php
buffer内存区域。

开启缓存后,脚本输出的内容存入了输出缓存中
,这时没有处于等待输出状态的内容,你直接使用flush()不会向客户端发出任何内容。而
ob_flush()的作用就是将本来存在输出缓存中的内容取出来,设置为等待输出状态,但不会直接发送到客户端
,这时你就需要先使用
ob_flush()再使用flush(),客户端才能立即获得脚本的输出。

php默认开启output_buffering机制,只不过,通过调用ob_start()函数据output_buffering值扩展到足够大。也可以指定$chunk_size来指定output_buffering的值。$chunk_size默认值是0,表示直到脚本运行结束,php
buffer中的数据才会发送到浏览器。如果你设置了$chunk_size的大小,则表示只要buffer中数据长度达到了该值,就会将buffer中的数据发送给浏览器。

一. flush和ob_flush的正确顺序,正确应是,先ob_flush再flush,如下: ob_flush();
flush();
如果Web服务器的操作系统是windows系统,那顺序颠倒或者不使用ob_flush()也不会出现问题。[有待求证
] 但是在Linux系统上就无法刷新输出缓冲。

当然,你可以通过指定$ouput_callback,来处理buffer中的数据。比如函数ob_gzhandler,将buffer中的数据压缩后再传送给浏览器。

output buffering函数 1.bool ob_start ([ callback $output_callback [, int $chunk_size
[, bool $erase ]]] )
激活output_buffering机制。一旦激活,脚本输出不再直接出给浏览器,而是先暂时写入php
buffer内存区域。
php默认开启output_buffering机制,只不过,通过调用ob_start()函数据output_buffering值扩展到足够

。也可以指定$chunk_size来指定output_buffering的值。$chunk_size默认值是0,表示直到脚本运行结束,php
buffer中的数据才会发送到浏览器。如果你设置了$chunk_size的大小
,则表示只要buffer中数据长度达到了该值,就会将buffer中
的数据发送给浏览器。
当然,你可以通过指定$ouput_callback,来处理buffer中的数据。比如函数ob_gzhandler,将buffer中的数据压缩后再传送给浏览器。
第三个参数:是否擦除缓存,可选,默认是true,如果设置为false,则在脚本执行结束前,缓存都不会被清除。
2.ob_get_contents
获取一份php
buffer中的数据拷贝。值得注意的是,你应该在ob_end_clean()函数调用前调用该函数,否则ob_get_contents()返回一个空字符中。

2.ob_get_contents
获取一份php
buffer中的数据拷贝。值得注意的是,你应该在ob_end_clean()函数调用之前调用该函数,否则ob_get_contents()返回一个空字符中。

可以使用ob_get_contents()以字符串形式获取服务端缓存的数据,
使用ob_end_flush()则会输出被缓存起来的数据,并关闭缓存。
而使用ob_end_clean()则会静默的清除服务端缓存的数据,而不会有任何数据或其他行为。
服务端的缓存是堆叠起来的,也就是说你在开启了ob_start()后,关闭之前,在其内部还
可以开启另外一个缓存ob_start()。

3.ob_end_flush与ob_end_clean
这二个函数有点相似,都会关闭ouptu_buffering机制。但不同的是,ob_end_flush只是把php
buffer中的数据冲(flush/send)到客户端浏览器,而ob_clean_clean将php
bufeer中的数据清空(erase),但不发送给客户端浏览器。ob_end_flush调用之后,php
buffer中的数据依然存在,ob_get_contents()依然可以获取php
buffer中的数据拷贝。而ob_end_clean()调用之后ob_get_contents()取到的是空字符串,同时浏览器也接收不到输出,即没有任何输出。

不过你也要务必保证关闭缓存的操作和开启缓存的操作数量一样多。
ob_start()
可以指定一个回调函数来处理缓存数据,如果一个ob_start()内部嵌套了另一个ob_start(),我们假定,外层的ob_start(),编号是A,内层的ob_start()编号是B,它们各自制定了一个回调函数分别是functionA和functionB,那么在缓存B中的数据输出时,它会先辈funcitonB回调函数处理,再交给外层的functionA回调函数处理,之后才能输出到客户端。

惯用案例
常常在一些模板引擎和页面文件缓存中看到ob_start()使用。下面湿CI中加载模板的程序代码:
复制代码 代码如下:
  /*
   * Buffer the output
   *
   * We buffer the output for two reasons:
   * 1. Speed. You get a significant speed boost.
   * 2. So that the final rendered template can be
   * post-processed by the output class.  Why do we
   * need post processing?  For one thing, in order to
   * show the elapsed page load time.  Unless we
   * can intercept the content right before it’s sent to
   * the browser and then stop the timer it won’t be accurate.
   */
  ob_start();
  // If the PHP installation does not support short tags we’ll
  // do a little string replacement, changing the short tags
  // to standard PHP echo statements.
  if ((bool) @ini_get(‘short_open_tag’) === FALSE AND
config_item(‘rewrite_short_tags’) == TRUE)
  {
                        //替换短标记=***>
   echo eval(‘?>’.preg_replace(“/;*s*?>/”, “; ?>”,
str_replace(‘=’, ‘  // Return the file data if requested
  if ($_ci_return === TRUE)
  {
   $buffer = ob_get_contents();
   @ob_end_clean();
   return $buffer;
  }
  /*
   * Flush the buffer… or buff the flusher?
   *
   * In order to permit views to be nested within
   * other views, we need to flush the content back out whenever
   * we are beyond the first level of output buffering so that
   * it can be seen and included properly by the first included
   * template and any subsequent ones. Oy!
   *
   */
  if (ob_get_level() > $this->_ci_ob_level + 1)
  {
   ob_end_flush();
  }
  else
  {
                        //将模板内容添加到输出流中
   $_ci_CI->output->append_output(ob_get_contents());
                        //清除buffer
   @ob_end_clean();
  }

另外,手册说,对于某些web服务器,比如apache,在使用回调函数有可能会改变程序当前的工作目录,解决方法是在回调函数中自行手动把工作目录修改回来,用chdir函数,这点似乎不常遇到,遇到的时候记得去查手册吧。

3.ob_end_flush与ob_end_clean 这二个函数有点相似,都会关闭ouptu_buffering机制。但不同的是,ob_end_flush只是把php
buffer中的数据冲(flush/send)到客户端浏览器,而ob_clean_clean将php
bufeer中的数据清空(erase),但不发送给客户端浏览器。

ob_end_flush调用之前 ,php
buffer中的数据依然存在,ob_get_contents()依然可以获取php
buffer中的数据拷贝。

而ob_end_flush()调用之后
ob_get_contents()取到的是空字符串,同时浏览器也接收不到输出,即没有任何输出。

可以使用ob_get_contents()以字符串形式获取服务端缓存的数据,使用ob_end_flush()则会输出被缓存起来的数据,并关闭缓存。
而使用ob_end_clean()则会静默的清除服务端缓存的数据,而不会有任何数据或其他行为。
服务端的缓存是堆叠起来的,也就是说你在开启了ob_start()后,关闭之前,在其内部还可以开启另外一个缓存ob_start()。不过你也要务必保证关闭缓存的操作和开启缓存的操作数量一样多。
ob_start()
可以指定一个回调函数来处理缓存数据,如果一个ob_start()内部嵌套了另一个ob_start(),我们假定,外层的ob_start(),编号是A,内层的ob_start()编号是B,它们各自制定了一个回调函数分别是functionA和functionB,那么在缓存B中的数据输出时,它会先辈funcitonB回调函数处理,再交给外层的functionA回调函数处理,之后才能输出到客户端。

发表评论

电子邮件地址不会被公开。 必填项已用*标注