PHP fsockopen 异步调用接口在nginx上偶尔实效的情况
private function fsock_asy_do($get){
$fp = fsockopen("ssl://www.xxx.com", 443, $errno, $errstr, 30);
if (!$fp) {
echo "$errstr ($errno)<br />\n";
} else {
stream_set_blocking($fp,0);//开启非阻塞模式
$out = "GET /".$get." HTTP/1.1\r\n";
$out .= "Host: www.xxxx.com\r\n";
$out .= "Connection: Close\r\n\r\n"; fwrite($fp, $out);
/*忽略执行结果
while (!feof($fp)) {
echo fgets($fp, 128);
}*/
usleep(1000);
fclose($fp);
}
}
在这段代码里面,调用后忽略执行结果直接返回,可用于php异步执行。
在nginx服务器上有一个比较诡异的情况就是有时候无法调用异步的脚本。
查阅相关资料后,是nginx 499 的问题。
其中解决方案有以下,经过对每一个方案的验证最终得出结果:
1、NGINX 499
查看 NGINX access log,发现这样的请求会以 499
(Client Closed Request)记录。确定问题是因为:客户端主动端口请求连接时,NGINX 不会将该请求代理给上游服务(FastCGI PHP 进程),这个时候 access log 中会以 499 记录这个请求。
要解决这个问题需要将 NGINX FastCGI 忽略客户端中断配置打开:
fastcgi_ignore_client_abort on;
这样无论客户端是否断开,都会将这个请求代理给上游,并且会记录上游服务处理后的返回状态。
2、NGINX 线程原因
将nginx的worker_processes 由之前的auto修改为2(我的是单核服务器)
3、NGINX 499
nginx对499的定义是”client has closed connection”,并且在这些情况下会返回这个状态码:
- upstream 在收到读写事件处理之前时发现连接不可用。
- server处理请求未结束,而client提前关闭了连接。
- upstream出错,执行next_upstream时发现连接不可用。
一个不安全的做法是在fclose之前,让当前的进程先睡眠一段时间;我这里设置为10毫秒,这10毫秒的延迟对我完成整个请求的影响不大,同时我也认为nginx一定能在10毫米内把请求转到fastcgi去执行。这个时间间隔很难把握,不能保证php一定有执行到。
这种方式并不是真正的异步,只是很取巧的强制关闭连接而不等待服务器端响应。所以在Laruence的那2篇文章中,有2个问题:
- PHP使用fsock不能叫做异步,只是伪异步。
- fwrite之后马上执行fclose,nginx会直接返回499
由于我的代码上面usleep为1000,初步估计是时间不够导致没发出去就close了,所以调整为20000。并进行最后测试。
结果晚点更新。
更新结果:测试2天结果显示正常,的确是usleep数值过小的问题。
最新文章
- node.js xtemplate的使用实例
- iOS如何跳到系统设置里的各种设置界面
- Qt Connect 信号 槽
- .Net程序员安卓学习之路4:使用xutils Get Post数据
- ArcEngine编写WebService
- BW:如何加载和生成自定义的层次结构,在不使用平面文件的SAP业务信息仓库
- js json与对象的相互转换
- Linux如何修改文件/文件夹内所有文件的权限
- 【原】centos6.5下cdh4.6 Oozie安装
- vmware workstation11虚拟机破解版(附安装教程) 32/64位
- Material风格的Quick组件,妈妈再也不用担心我的界面不好看了
- centos安装如何选择安装包
- Hibernate 、多表关联映射-多对一关系(many-to-one)
- python 调用图灵机器人api实现简单的人机交互
- 3步轻松搭建Unity3d安卓开发环境
- Redis线程模型
- Javascript var 和 let 的区别
- delegate的Invoke和BeginInvoke方法
- java.net.SocketException四大异常解决方案
- 三角形的优雅值(map和哈希表)