有时候,运行 Nginx、PHP-CGI(php-fpm) Web服务的 Linux 服务器,突然系统负载上升,使用 top 命令查看,很多 php-cgi 进程 CPU 使用率接近100%。后来,我通过跟踪发现,这类情况的出现,跟 PHP 的 file_get_contents() 函数有着密切的关系。
大、中型网站中,基于 HTTP 协议的 API 接口调用,是家常便饭。PHP 程序员们喜欢使用简单便捷的 file_get_contents("http://example.com/") 函数,来获取一个 URL 的返回内容,但是,如果http://example.com/这个网站响应缓慢,file_get_contents()
就会一直卡在那儿,不会超时。
我们知道,在 php.ini 中,有一个参数 max_execution_time 可以设置 PHP 脚本的最大执行时间,但是,在 php-cgi(php-fpm) 中,该参数不会起效。真正能够控制 PHP 脚本最大执行时间的是 php-fpm.conf 配置文件中的以下参数:
-
Thetimeout(inseconds)forservingasinglerequestafterwhichtheworkerprocesswillbeterminated
-
Shouldbeusedwhen'max_execution_time'inioptiondoesnotstopscriptexecutionforsomereason
-
'0s'means'off'
-
<valuename="request_terminate_timeout">0s</value>
默认值为 0 秒,也就是说,PHP 脚本会一直执行下去。这样,当所有的 php-cgi 进程都卡在 file_get_contents() 函数时,这台 Nginx+PHP 的 WebServer 已经无法再处理新的 PHP 请求了,Nginx 将给用户返回“502 Bad Gateway”。修改该参数,设置一个 PHP 脚本最大执行时间是必要的,但是,治标不治本。例如改成
<value name="request_terminate_timeout">30s</value>,如果发生 file_get_contents() 获取网页内容较慢的情况,这就意味着 150 个 php-cgi 进程,每秒钟只能处理 5 个请求,WebServer 同样很难避免“502 Bad Gateway”。
要做到彻底解决,只能让 PHP 程序员们改掉直接使用 file_get_contents("http://example.com/") 的习惯,而是稍微修改一下,加个超时时间,用以下方式来实现 HTTP GET 请求。要是觉得麻烦,可以自行将以下代码封装成一个函数。
-
<?php
-
$ctx=stream_context_create(array(
-
'http'=>array(
-
'timeout'=>1
-
)
-
)
-
);
-
file_get_contents("http://example.com/",0,$ctx);
-
?>
当然,导致 php-cgi 进程 CPU 100% 的原因不只有这一种,那么,怎么确定是 file_get_contents() 函数导致的呢?
首先,使用 top 命令查看 CPU 使用率较高的 php-cgi 进程。
top - 10:34:18 up 724 days, 21:01,3 users,load average: 17.86, 11.16, 7.69
Tasks: 561 total,15 running, 546 sleeping, 0 stopped, 0 zombie
Cpu(s):5.9%us,4.2%sy,0.0%ni, 89.4%id,0.2%wa,0.0%hi,0.2%si,0.0%st
Mem: 8100996k total,4320108k used,3780888k free, 772572k buffers
Swap:8193108k total,50776k used,8142332k free, 412088k cached
PID USERPRNIVIRTRESSHR S %CPU %MEMTIME+COMMAND
10747 www 18 0360m22m12m R 100.6 0.30:02.60 php-cgi
10709 www 16 0359m28m17m R 96.80.40:11.34 php-cgi
10745 www 18 0360m24m14m R 94.80.30:39.51 php-cgi
10707 www 18 0360m25m14m S 77.40.30:33.48 php-cgi
10782 www 20 0360m26m15m R 75.50.30:10.93 php-cgi
10708 www 25 0360m22m12m R 69.70.30:45.16 php-cgi
10683 www 25 0362m28m15m R 54.20.40:32.65 php-cgi
10711 www 25 0360m25m15m R 52.20.30:44.25 php-cgi
10688 www 25 0359m25m15m R 38.70.30:10.44 php-cgi
10719 www 25 0360m26m16m R7.70.30:40.59 php-cgi
找其中一个 CPU 100% 的 php-cgi 进程的 PID,用以下命令跟踪一下:
strace -p 10747
如果屏幕显示:
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
select(7, [6], [6], [], {15, 0})= 1 (out [6], left {15, 0})
poll([{fd=6, events=POLLIN}], 1, 0) = 0 (Timeout)
那么,就可以确定是 file_get_contents() 导致的问题了。
分享到:
相关推荐
有时候,运行 Nginx、PHP-CGI(php-fpm) Web服务的 Linux 服务器,突然系统负载上升,使用 top 命令查看,很多 php-cgi 进程 CPU 使用率接近100%。
服务器大量php-cgi.exe进程,导致CPU占用100%的解决[定义].pdf
防止连接数达到上限后 php-cgi 进程退出而导致服务终止。 修改配置 Web.ini: nginx_dir => 填写 nginx.exe 所在目录 php_cgi_dir => 填写 php-cgi.exe 所在目录 php_cgi_start => 填写 php-cgi 启动参数及端口 ...
Linux下php-cgi占内存100%电脑变的很卡的解决方法.docx
php-cgi为何占用CPU100%.pdf
HTMl(Hyper text mark up language), XML, CGI
CGI-LIB开发包,让C语言开发CGI容易起来,并在这个开发包上写了一些CGI程序。
cgi分析,很方便读取关键变量
Linux 下面进行网页交互通信cgi交叉编译必备工具链!
windows下 解决PHP-CGI 进程崩溃502的资源包,资源包中包括pthreadGC2.dll、registe.bat、restart_cgi.bat、RunHiddenConsole.exe、xxfpm.exe文件,详细使用方法可参考博客文章{windows下 解决PHP-CGI 进程崩溃502}
应用:DFA与DCCA计算 如何自定义下载:...
路由器Web管理界面,提供了中、英、日三种语言版本
主要介绍了linux php-cgi.exe占用cpu 100%的一次排障之旅,需要的朋友可以参考下
JAVA编写CGI程序手册
一、问题描述 1、19号手动升级到wordpress4.1,21号早上7点看网站php...4、top命令发现php-cgi 占用cpu很多,大部分时间都在80%到90%之间,cpu time+ 时间很长。 二、解决方法 1、百度搜索:php cgi cpu找到文章 http
linux 下的一个CGI 程序,含HTML程序文件。里面有一些注释。 非常适合处学者。 功能实现一个基于WEB页面的简单的乘法计算器
本安装文件修改了源码网的“CentOS 集成linux+php-cgi+nginx+mysql安装包“里面的部分文件,主要是修改了install.sh内的几个错误位置。目前我测试在centos 5.4下,安装一切正常! 使用这个安装之前请一定看一下本...
采用cgi实现动态网页。动态网页实现,基础,很好