本篇文章给大家分享了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('H:i:s')); }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=''){ $log_path = empty($path)?'./log_'.date('Y-m-d').'.log':$path; $time = date('Y-m-d H:i:s'); $error_page = 'http://' . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']; 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('H:i:s')); 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('H:i:s')); echo $re; }//这里省略了fib和setLog函数?>
这次再测试,就会发现浏览器会隔一段时间就收到一些相应,而不是之前的demo,需要脚本完全执行完才输出内容到客户端。同时,这个时候关闭客户端连接,服务器端当再次向客户端输出内容的时候,就会检查客户端连接已经断开了,这个时候脚本就会停止运行了。这是我们想要的测试结果。
4、再修改测试代码,这次不让一次输出一个很大的内容,而是有意操作缓冲区内容,让虽然不够从缓冲区输出到客户端的内容提前输出到客户端。
测试代码:
for ($i=0; $i < 10; $i++) { fib(25); setLog(date('H:i:s')); echo "hello " . date('H:i:s') . "<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主动断开连接,要使用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('H:i:s')); }function test(){ $size = ob_get_length(); header("content-length:" . $size); //header("connection:close"); ob_flush(); flush(); }//这里省略了fib和setLog函数?>
<完>
以上是http 浏览器主动断开连接 与 php主动断开连接的详细内容。更多信息请关注PHP中文网其他相关文章!