php 中断请求,http 浏览器主动断开连接 与 php主动断开连接

本篇文章给大家分享了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往客户端输出内容的时候判断的,那么我们把测试代码修改一下:

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') . "
";

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('H:i:s'));

}function test(){

$size = ob_get_length();

header("content-length:" . $size); //header("connection:close");

ob_flush();

flush();

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值