PHP settimelimit0长连接的实现分析

每个PHP脚本都限制了执行时间,所以我们需要通过 set_time_limit 来设置一个脚本的执行时间为无限长;然后使用 flush() 和 ob_flush() 来清除服务器缓冲区,随时输出脚本的返回值。

如下面这段脚本: 

<?php 
header("Content-Type: text/plain"); 
set_time_limit(0); 

$infoString = "Hello World" . "\n"; 
while( isset($infoString) ) 
{ 
echo $infoString; 
ob_flush(); 
flush(); 
sleep(5); 
} 
?> 


当我们执行后,每隔5秒钟,我们会得到一行 Hello World ,如果不按停止按钮,浏览器会不停的一行一行继续加载。

通过这一方法,我们可以完成很多功能,例如机器人爬虫、即时留言板等程序。 

注意事项: 有些服务器设置了gatway time out 。此时,过长时间的脚本运行会引起504错误超时, 这个错误只是网页端获取不到数据,但是服务器端的php脚本是任然在运行的。

    我已经试过因这个原因出过几次错。所以使用set_time_limit() 的时候要谨慎。引以为鉴。

下面的内容为转载: http://blog.sina.com.cn/s/blog_7193eeac01010imp.html

  

<?php
echo str_pad(" ",1024);
//当上面这句没有的时候浏览器没有任何输出 直到sleep函数设定的时间结束 才会输出
//原因如下面截图
for ($i=10; $i>0; $i--)
{
    echo $i.'<br />';
    ob_flush();
    flush();
    sleep(1);
}
ob_end_flush();
?>

flush和ob_flush的使用上有一些特别容易犯错的地方,造成无法刷新输出缓冲。
一. flush和ob_flush的正确顺序,正确应是,先ob_flush再flush,如下:
ob_flush();
flush();
如果Web服务器的操作系统是windows系统,那顺序颠倒或者不使用ob_flush()也不会出现问题。但是在Linux系统上就无法刷新输出缓冲。

二. 使用ob_flush()前,确保前面的内容大小足够4069字符。
一些Web服务器的output_buffering默认是4069字符或者更大,即输出内容必须达到4069字符服务器才会flush刷新输出缓冲,为了确保flush有效,最好在ob_flush()函数前有以下语句:
print str_repeat(" ", 4096);
以确保到达output_buffering值。

以下是代码片断:
for ($i=10; $i>0; $i--)
{
    echo $i.'<br />';
    ob_flush();
    flush();
    sleep(1);
}
ob_end_flush();

刷新PHP程序的缓冲,而不论PHP执行在何种情况下(CGI ,web服务器等等)。该函数将当前为止程序的所有输出发送到用户的浏览器。

flush() 函数不会对服务器或客户端浏览器的缓存模式产生影响。因此,必须同时使用 ob_flush() 和flush() 函数来刷新输出缓冲。

个别web服务器程序,特别是Win32下的web服务器程序,在发送结果到浏览器之前,仍然会缓存脚本的输出,直到程序结束为止。

有些Apache的模块,比如mod_gzip,可能自己进行输出缓存,这将导致flush()函数产生的结果不会立即被发送到客户端浏览器。

甚至浏览器也会在显示之前,缓存接收到的内容。例如 Netscape 浏览器会在接受到换行或 html 标记的开头之前缓存内容,并且在接受到 </table> 标记之前,不会显示出整个表格。

一些版本的 Microsoft Internet Explorer 只有当接受到的256个字节以后才开始显示该页面,所以必须发送一些额外的空格来让这些浏览器显示页面内容。

下面是另外一篇转载文章: http://www.nqhua.com/2011/09/23/226.html

ob_start,flush,ob_flush
for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    flush();
    sleep(1);
 }

有了解过PHP缓存输出控制函数的朋友肯定对上面这段代码很熟悉,它想实现的效果是每个1秒输出1个数字,完成全部输出需要10秒,不过实际执行中你会发现奇怪的现象,有些人或者有些时候它的表现如你所愿,而有些人或者有些时候却是10秒后才会一次性输出10个数字。我曾经为此抓狂不已,有朋友留言说这个情况往往是因为IE的缓存必须达到256个字符才会输出,可实际上我之前也考虑到IE的情况,可依旧会有时灵时不灵的情况。今天仔细读过手册才明白,这些不可预料的现象是有它的理由的。

原来php.ini中有两个关键参数会影响到php的缓存输出控制:

参数1:output_buffering :on/off 或者整数。设置为on时,将在所有脚本中使用输出缓存控制,不限制缓存的大小。而设置为整数时,如output_buffering=4096,当缓存数据达到4096字节时会自动输出刷新缓存。而这个参数的不同正是导致以上代码在不同时候执行结果不同的原因。当output_buffering关闭时,脚本所有的输出(echo)都会即时发送到客户端,执行上面代码时就是每秒输出一个数字。而开启output_buffering后,输出内容就会先缓存在服务端,直到脚本结束时才一起发送给客户端。

参数2:implicit_flush:on/off。设定ON意味着,当脚本有输出时,自动立即发送到客户端。相当于在echo后自动加flush()。

 

php缓存输出控制的相关函数:

bool ob_start ([ callback $output_callback [, int $chunk_size [, bool $erase ]]] )

第一个参数:回调函数,可选。在缓存输出前可以对其进行过滤或其他处理。最常见的用法是ob_start(‘ob_gzhandler’),即对缓存的数据进行gzip压缩后再发送给客户端。
第二个参数:缓存块的大小,可选。如果被缓存的内容达到或操作缓存块的大小,缓存会自动输出。默认值是0,指不限定大小,缓存到结束为止。还有个特殊值1,代表chunk_size=4096。
第三个参数:是否擦除缓存,可选,默认是true,如果设置为false,则在脚本执行结束前,缓存都不会被清除。

可以使用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回调函数处理,之后才能输出到客户端。

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

flush()和ob_flush()

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

反复研究了手册的说明,参考了手册中一些人的留言,自己琢磨应该是这样的:

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

也就是说本文开头的脚本,可以根据缓存开启与否,有如下几种不同的写法:

注:以下代码都未考虑IE缓存必须大于256字节才输出的问题,如在IE下测试,请在代码开始加一句:“echo str_repeat(‘ ‘,4096)”。

写法一:

output_buffering = off
 implicit_flush=off
 
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    flush();
    sleep(1);
 }

写法二:

output_buffering = on
 implicit_flush=off
 
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    ob_flush();
    flush();
    sleep(1);
 }

写法三:

output_buffering = off
 implicit_flush=off
 
 ob_start();
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    ob_flush();
    flush();
    sleep(1);
 }

写法四: 

output_buffering = on
 implicit_flush=off
 
 ob_end_flush();
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    flush();
    sleep(1);
 }

写法五:

output_buffering = on
 implicit_flush=off
 
 ob_end_clean();
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    flush();
    sleep(1);
 }

写法六:

output_buffering = on;
 implicit_flush=on
 
 ob_end_clean();
 // 或者ob_end_flush();
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    sleep(1);
 }

写法七:

output_buffering = on;
 implicit_flush=on
 
 ob_end_clean();
 // 或者ob_end_flush();
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    flush();
    sleep(1);
 }

写法八:

output_buffering = off
 implicit_flush=on
 
 for($i=0;$i<10;$i++) {
    echo $i.'<br />';
    sleep(1);
 }

 

 

 

原文地址:https://www.cnblogs.com/zjfazc/p/2587989.html