PHPer 肯定收到过这样的投诉:小菊花一直在转!你们网站怎么这么卡!当我们线上业务遇到这种卡住(阻塞)的情况,大部分 PHPer 会两眼一抹黑,随后想起那句名言:性能瓶颈都在数据库然后把锅甩给DBA,赶紧找找慢sql,但这是非常错误的做法,因为有太多因素能导致业务卡住,下面列举几种常见的卡住问题。

1.死循环

最常见的就是写出了死循环代码

  1.  <?php
    while(1){
    //do something
    if($condition){
    //满足条件后退出循环
    break; } }

上述代码通过$condition控制循环退出,如果程序验证不严格,某些情况$condition永远为真就会导致请求卡死。

2.sesstion_start函数导致卡死

PHP的session锁等待(ps:很多地方叫做session死锁,这不太符合死锁定义),这个相信大部分PHPer都遇到过,PHP默认会把session信息存储在/tmp/sess_下面的session文件里面,调用session_start()函数的时候会调用flock系统调用给session文件加锁,如果前一个请求没有结束或者手动释放session就会导致后面的请求无法获得锁,卡死在session_start()这个地方。下面举个例子,比如这种代码:

  1.  setInterval(function () {
    $.post("/ajax/doSomething", {}, function (result) {//1s进行一次ajax
    });
    }, 1000)//1000ms == 1s

前端js定时通过ajax请求一下后端PHP的接口(/ajax/doSomething)做一些比较耗时的事情,写代码的人可能想当然的认为第一次的请求即使没有处理完,也不会影响第二次的请求,因为有很多的FPM进程每次请求会分发到不通的进程,但殊不知第二次请求会卡死在session_start()

3.flock函数导致卡死

最常见的场景就是写日志,在PHP代码中确保每次fwrite写的日志内容小于8k的情况下我们可以利用append原子追加方式写日志,但是如果保证不了小于8k我们就需要在每次写日志前给文件加文件锁来避免两次日志间产生穿插的情况,代码如下:

 <?php
$fp = fopen("/home/guoxinhua/php.log", "a+");
if (flock($fp, LOCK_EX)) { //给日志文件加锁
//do something
fwrite($fp, "the huge string\n");
flock($fp, LOCK_UN); // 释放锁定
}

如果在A进程获得锁后由于某种问题阻塞了那么B进程就会卡死在第三行flock的位置,除非A进程被kill掉,系统会自动释放这个文件锁

注意还有很多其他类型的锁即使进程被kill也不会自动被释放。
这个8k是可以改的,和glibc中的fwrite很多细节也不一样.

4. 网络客户端未设置超时时间

MySQL、CURL、Swoole\Client 等网络客户端未设置超时可能会导致进程阻塞。Swoole\Client 建立 TCP 连接的时候connect方法的最后一个参数是超时时间,-1即为永不超时,注意这里设置不是单指这次connect方法,而是后面所有的send,recv都永不超时,在同步阻塞的编程模式下,如果此时对端机器直接宕机等原因导致网络不通,那么本端业务的表现就是卡死状态,所有的send,recv方法都将被阻塞,代码如下:

 <?php

 $cli = new Swoole\Client(SWOOLE_SOCK_TCP);

 if ($cli->connect('127.0.0.1', 9501,-1)) {

 $cli->send("data");

 $cli->recv();

 } else {

 echo "connect failed.";

 }

5. Swoole协程的lock

在 Swoole 协程模式下,不正确的使用lock也会导致所有协程大面积卡死,如下代码,通过go方法创建2个协程(不理解协程的同学可以理解为创建了2个线程),第一个协程lock获得锁后在co::sleep位置让出了cpu此时开始执行第二个协程,第二个协程会卡死在第6行获得锁的位置,同时第一个协程也永远无法恢复继续执行。

 <?php

 $lock = new Swoole\Lock();

 $c = 2;//创建2个协程

 while ($c--) {

 go(function () use ($lock) {//创建协程

 $lock->lock();//获得锁

 Co::sleep(1);//让出cpu

 $lock->unlock();//释放锁

 });

 }

如何发现卡死

上述只是举了一些例子,真实业务中还有各种姿势的卡死,遇到这种问题有经验的PHPer会用strace -p命令查看当前PHP进程到底阻塞在哪个系统调用上面来定位问题,但这种方式有几个问题:

  • 定位问题不清晰
    比如死锁这种问题strace的时候只能看到类似futex(0x7f4c8d567128, FUTEX_WAIT, 2, NULL)这种信息,非常的不直观,很多人根本不知道哪些PHP代码会触发futex系统调用,还有前文提到session_start那个问题,很多人根本不知道这里会触发flock,也就说很难根据一个系统调用定位到具体问题。
  • 不知道-p哪一个进程
    我们线上环境通常会启动几十个甚至上百个PHP进程,在有些请求卡死,有些请求正常的情况下,你到底该strace -p哪个进程呢?貌似只能碰碰运气了。
  • 发现不了死循环的问题
    由于strace命令的原理是追踪所有的系统调用,如果是前文提到的第一种情况,也就是死循环的卡死,strace根本无法获得任何有用的信息。此时我们只能用gdb工具来获取当前死循环在哪里具体,具体做法如下:首先:gdb attach后面接个进程id。
    然后:
    p (char *)executor_globals.current_execute_data.func.op_array.filename.val打印当前执行的PHP文件。
    p (char *)executor_globals.current_execute_data.func.op_array.function_name.val打印当前执行的函数名。
    p executor_globals.current_execute_data.opline.lineno打印当前执行的行数。
    进一步也可以获取调用堆栈这里就不展开了。
    但这明显太底层了,很多细节要注意,不精通PHP内核的人很难这样找问题(ps:通过.gdbinit能稍微减少点难度,但是也有很多其他问题)。

使用 Swoole Tracker 发现卡死问题

针对上述问题,Swoole官方出了一个解决方案 Swoole Tracker 的堆栈工具,同时支持FPM和Swoole。
使用方法很简单:

    • 首先点击上面的连接注册个账户。
    • 然后装上swoole_tracker扩展。
    • 最后登陆后台,在调试器=>进程列表中点击堆栈按钮就能获得当前卡在哪了,如图:
    • 结尾

      除了上面的卡死问题,还有一种情况是调用变慢,比如原来一个系统调用5ms,但是由于网络等等原因,这个调用100ms才返回,业务的表现是变慢了而不是卡死在那里,这种情况通过tracker的抓堆栈工具是无法定位问题的,因为卡住时间很短,很难抓到调用堆栈,此时需要Swoole工具链中的另外一个工具阻塞IO检测工具我们会在后面给大家介绍。

教你使用 Swoole-Tracker 秒级定位 PHP 卡死问题的更多相关文章

  1. PHP 教你使用 Swoole-Tracker 秒级定位 PHP 卡死问题

    PHPer 肯定收到过这样的投诉:小菊花一直在转!你们网站怎么这么卡!当我们线上业务遇到这种卡住(阻塞)的情况,大部分 PHPer 会两眼一抹黑,随后想起那句名言:性能瓶颈都在数据库然后把锅甩给DBA ...

  2. 基于Flink秒级计算时CPU监控图表数据中断问题

     基于Flink进行秒级计算时,发现监控图表中CPU有数据中断现象,通过一段时间的跟踪定位,该问题目前已得到有效解决,以下是解决思路:   一.问题现象       以SQL02为例,发现本来10秒一 ...

  3. CCKiller:Linux轻量级CC攻击防御工具,秒级检查、自动拉黑和释放 《CCKiller:Linux轻量级CC攻击防御工具,秒级检查、自动拉黑和释放》来自张戈博客

    张戈博客很久以前分享过一个CC攻击的防御脚本,写得不怎么样,不过被51CTO意外转载了.博客从此走上了经常被人拿来练手的不归之路. 当然,还是有不少朋友在生产环境使用,并且会留言询问相关问题.根据这些 ...

  4. 软件架构自学笔记-- 转载“虎牙在全球 DNS 秒级生效上的实践”

    虎牙在全球 DNS 秒级生效上的实践 这次分享的是全球 DNS 秒级生效在虎牙的实践,以及由此产生的一些思考,整体上,分为以下 5 各部分: 背景介绍: 方案设计和对比: 高可用: 具体实践和落地: ...

  5. php Swoole实现毫秒级定时任务

    项目开发中,如果有定时任务的业务要求,我们会使用linux的crontab来解决,但是它的最小粒度是分钟级别,如果要求粒度是秒级别的,甚至毫秒级别的,crontab就无法满足,值得庆幸的是swoole ...

  6. Swoft 2.0.5 更新,新增高效秒级定时任务、异常管理组件

    什么是 Swoft ? Swoft 是一款基于 Swoole 扩展实现的 PHP 微服务协程框架.Swoft 能像 Go 一样,内置协程网络服务器及常用的协程客户端且常驻内存,不依赖传统的 PHP-F ...

  7. 《DNS稳定保障系列3--快如闪电,域名解析秒级生效》

    在刚刚过去的双十一,又是一个全民狂欢的盛宴,天猫双十一的成交量高达2684亿.无数小伙伴在淘宝.天猫里买买买,今年你又剁手了多少?言归正传,在你疯狂秒杀的时候,有没有发现,今年的购物体验一如既往的好, ...

  8. 虎牙在全球 DNS 秒级生效上的实践

    本文整理自虎牙中间件团队在 Nacos Meetup 的现场分享,阿里巴巴中间件受权发布. 这次分享的是全球 DNS 秒级生效在虎牙的实践,以及由此产生的一些思考,整体上,分为以下5各部分: 背景介绍 ...

  9. Swoole实现毫秒级定时任务

    项目开发中,如果有定时任务的业务要求,我们会使用linux的crontab来解决,但是它的最小粒度是分钟级别,如果要求粒度是秒级别的,甚至毫秒级别的,crontab就无法满足,值得庆幸的是swoole ...

随机推荐

  1. python_正则表达式学习

    re.match()函数: 函数语法: re.mathch ( pattern , string , flags = 0) 参数说明: pattem 匹配的正则表达式 string 要匹配的字符串 f ...

  2. 多个浏览器下应用前端JS实现一键导出excel表

    自己试验了几种方法,找到一种较为全面的一种方式一键输出Excel表格,代码如下 <!DOCTYPE html> <html> <head lang="en&qu ...

  3. kali linux查看局域网下所有IP,并对指定IP实施局域网内攻击(断网,随时查看对方密码,上网痕迹等)

    首先我们打开我们熟悉的kali linux操作系统,利用指令: ifconfig 来确认本机的ip地址 确认了本机的ip地址之后,利用一下的指令查看局域网下所有ip: fping -g 本机IP地址/ ...

  4. ICMP重定向 Redirect netwox libpcap netwag

    搭建环境 两台虚拟机. 攻击者:192.168.1.8 被攻击者:192.168.1.9 网络配置 主机均采用DHCP 如果没有路由器,可以使用NAT模式来做 攻击者配置 打开转发数据包功能 # su ...

  5. 一次Ubuntu16系统的找回root密码的过程

    一 背景知识介绍 ubuntn系统的默认配置,超级用户root的密码是随机的,如果没提前对root用户的密码进行设置, 是不能直接通过root的身份来进行某些操作的,必须使用sudo命令,并通过普通用 ...

  6. ES新提案:双问号操作符

    摘要: 简单实用的新特性. 原文:ES新提案:双问号操作符 译者:前端小智 本文主要讲Gabriel Isenberg撰写的ES提案"Nullish coalescing for JavaS ...

  7. python 中文分词库 jieba库

    jieba库概述: jieba是优秀的中文分词第三方库 中文文本需要通过分词获得单个的词语 jieba是优秀的中文分词第三方库,需要额外安装 jieba库分为精确模式.全模式.搜索引擎模式 原理 1. ...

  8. 更改docker默认网段

    #本文档旨在说明创建docker时注意的事项:我们在局域网中使用Docker,最常遇到的一个困惑,就是有时候跨网段结果出现网络不通.原因是因为Docker默认生成的网关和我们的局域网网段有时候是冲突的 ...

  9. 转载-logbock.xml

    转载:CS408 Spring boot——logback 基础使用篇:https://www.cnblogs.com/lixuwu/p/5804793.html#autoid-0-0-0 阅读目录 ...

  10. windowsServer ------ 安装IIS

    1.找到服务器管理器,点击添加角色,一步步执行 2.添加IIS 相关组件 勾选web服务器 下一步 将web服务iis 相关组件全部勾选,ftp 可不选 选择好后安装 等一会 关闭 可以查看到所安装角 ...