首页  >  文章  >  后端开发  >  http 浏览器主动断开连接 与 php主动断开连接

http 浏览器主动断开连接 与 php主动断开连接

不言
不言原创
2018-04-26 14:41:214962浏览

本篇文章给大家分享了http 浏览器主动断开连接 与 php主动断开连接,有兴趣的朋友可以看一看

摘要:事件起因是因为平时在开发中遇到的疑惑。一次是浏览器客户端主动断开了连接后,发现服务器端的php脚本还在执行,以至于不知道怎样让脚本停下来。还有一次是有需求让php脚本主动断开连接,然后后续脚本继续执行(一个耗时任务),所以有了这篇博客。

一、浏览器主动断开连接

  在常用的LAMP组合下,我们认为,浏览器访问一个php脚本,脚本开始执行,脚本输出内容,并结束运行,apache响应http,浏览器收到http响应,显示结果。
  下来考虑下特殊的情况。
  1、浏览器发送http请求,php执行了一个耗时任务(20s)(假设php的set_time_limit设置的是30s),在此期间浏览器无响应,用户点击浏览器X,浏览器主动断开连接,php脚本是否还继续运行。
  假设耗时任务是:计算fib(25),浏览器测试响应需要时间1.15s,每执行一次耗时任务,写文件Log写一次,执行10次耗时任务,在执行第5次的时候,客户端主动断开连接,观察情况。
代码如下:

<?phpfor ($i=0; $i < 10; $i++) { 
    fib(25);
    setLog(date(&#39;H:i:s&#39;));
}function fib($n = 3){
    if($n == 0){        return 1;
    }    if($n == 1){        return 1;
    }    return fib($n - 1) + fib($n -2);
}function setLog( $massage, $path=&#39;&#39;){
    $log_path = empty($path)?&#39;./log_&#39;.date(&#39;Y-m-d&#39;).&#39;.log&#39;:$path;    $time = date(&#39;Y-m-d H:i:s&#39;);    $error_page = &#39;http://&#39; . $_SERVER[&#39;HTTP_HOST&#39;] . $_SERVER[&#39;REQUEST_URI&#39;];
    file_put_contents($log_path, "LOG TIME:".$time.PHP_EOL, FILE_APPEND);
    file_put_contents($log_path, "LOG URL:".$error_page.PHP_EOL, FILE_APPEND);    if(is_array($massage)){        $massage = json_encode($massage);
    }
    file_put_contents($log_path, "LOG MESSAGE:".$massage.PHP_EOL.PHP_EOL, FILE_APPEND);
}?>

  浏览器在执行到5.44s的时候断开了连接。
  日志显示:脚本执行完了10次循环。
  这与我们之前认为的不一样;

  2、优化一下,看到网上说,php判断客户端连接是否断开,是在php往客户端输出内容的时候判断的,那么我们把测试代码修改一下:

<phpfor ($i=0; $i < 10; $i++) { 
    fib(25);    setLog(date(&#39;H:i:s&#39;));    echo "hello";
}

//这里省略了fib和setLog函数
?>

  再次测试一下,发现和上一次的测试结果是一样的。究其原因:php往客户端输出内容的时候,要有3个缓冲阶段,分别是:
  php buffer => web server buffer => browser buffer
  只有当缓冲区满了的时候才会输出到客户端,这其实就是,后端每隔一段时间输出内容到前端的原理。当然也是可以控制当缓冲区没有满的时候,也让输出到客户端。

  3、再修改测试代码,让输出客户端的内容足够大:

<?php$re = "";for($i=0; $i < 10000; $i++){    $re .= "aa";
}for ($i=0; $i < 10; $i++) { 
    fib(25);
    setLog(date(&#39;H:i:s&#39;));    echo $re;
}//这里省略了fib和setLog函数?>

  这次再测试,就会发现浏览器会隔一段时间就收到一些相应,而不是之前的demo,需要脚本完全执行完才输出内容到客户端。同时,这个时候关闭客户端连接,服务器端当再次向客户端输出内容的时候,就会检查客户端连接已经断开了,这个时候脚本就会停止运行了。这是我们想要的测试结果。

  4、再修改测试代码,这次不让一次输出一个很大的内容,而是有意操作缓冲区内容,让虽然不够从缓冲区输出到客户端的内容提前输出到客户端。
测试代码:

for ($i=0; $i < 10; $i++) { 
    fib(25);    setLog(date(&#39;H:i:s&#39;));    echo "hello " . date(&#39;H:i:s&#39;) . "<br>";
    ob_flush();
    flush();
}
//这里省略了fib和setLog函数

小结:

  • 原则上客户端主动断开连接,php脚本即停止运行;

  • 但是前提是php知道客户端断开连接是怎么知道的,只有当php输出内容到客户端(不是php缓冲区、不是web server缓冲区),php才知道客户端连接中断了,才会停止运行;

  • php输出内容到客户端,有两种方式。一是填满内容到缓冲区自动发送到客户端;二是使用ob_flush,flush函数主动将缓冲区内容冲刷给客户端;

  • php脚本运行还受到内部的脚本计时器限制,可以在php.ini或者宿主apache配置文件中配置,或者脚本中通过set_time_limt函数设定;

  • 当客户端主动断开连接,而php脚本没有停止运行的时候,还要受限制于脚本计时器;

  • 当php脚本设置ignore_user_abort(true); 则即使客户端连接断开,且php输出内容到客户端知道了客户端连接断开,也不会停止脚本执行;

  • php内部,系统维护的连接状态,可以通过函数connection_status的返回值检查,0 : normal; 1 : aborted(断开连接); 2 : timeout; 改状态的检测也是需要php脚本输出内容到客户端才会知道,否则一直都是0;

  • 另外还有一个函数也可以检测客户端连接是否断开(connection_aborted),0正常,1断开。

  • 有个奇怪的问题是,当客户端连接已经断开,php脚本输出两次后,状态位才变成1;

二、php服务器端主动断开连接

  要让php主动断开连接,要使用http响应头里面的content-length和connection两个字段,意义分别为:

  • content-length,当客户端收到的响应头content-length,则当相应体收到指定大小后,就会断开与服务器的连接;

  • connection,当客户端收到响应头connection的值为close或者keep-alive,决定关闭当前tcp连接或者继续使用当前连接作下一次请求;

  • 测试发现,当只指定conetent-length的时候也能达到php主动断开连接;

  • 其实说是php主动断开连接,其实是php通知客户端主动断开的连接;

示例代码:

<?phpecho "hello world";

test();for ($i=0; $i < 10; $i++) { 
    fib(25);
    setLog(date(&#39;H:i:s&#39;));
}function test(){
    $size = ob_get_length();
    header("content-length:" . $size);    //header("connection:close");
    ob_flush();
    flush();
}//这里省略了fib和setLog函数?>



ba33df4ef3efea3726e09756c08771e7



以上是http 浏览器主动断开连接 与 php主动断开连接的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn