写给 PHP 程序员的信号处理教程

新闻 前端
信号相关的知识点其实有很多,还需要继续深入研究~上文中的PHP源码为7.1.25版本,各个版本可能不太一样。

“ 今天晚上吃卤煮,领桌的妹子问我,这玩意儿能吃么?我:你觉得能吃就能吃。。。和内容无关的主题 

01

什么是信号

信号就是事件发生时,对进程的一种通知机制(也叫软件中断)。当一个进程收到信号后,内核会暂停该进程正在执行的代码,并跳转到对应的信号处理函数中,如果处理函数不中断,执行完处理函数后,会继续执行之前中断的地方往下执行。

我们在FPM模式下写代码,不会遇到信号处理相关的问题,但是CLI模式下一些常驻内存的脚本,如何能够自由的重启、关闭、退出前做一些清理工作(断开链接,删除临时文件等)?

02

C的信号处理举例

上图中,我为信号SIGINT注册了处理函数sigint_handle,捕获到信号后,输出内容后退出,简单易懂吧。执行  gcc -o run run.c && ./run ,然后 CTRL+C(会触发SIGINT信号) ,成功输出: 成功捕获到信号2! ,程序直接结束运行。

03

PHP的信号处理举例

pcntl_signal是PHP的信号处理注册方法,上面实现的功能和C实现的基本一致,不同的是,当前进程不会退出,并且多输出了一个signinfo(PHP是C写的,为啥刚刚C语言的没有信号相关的信息呢?因为PHP使用的是另一个信号函数 sigaction, 有兴趣的可以了解一下 )

04

PHP的信号处理并不是直接调用C

这个是pcntl初始化的时候,将pcntl_signal_dispatch注册为tick的处理函数

pcntl_signal会将处理函数放到信号集合中(PHP的hash table),而php_signale4最终会调用sigaction进行底层的信号管理。

这里我省略了大量代码,将关键的点标记了出来,其实PHP维护一个自己的信号集合,每当调用 pcntl_signal_dispatch时就会查询是否有信号,上面的SIG_BLOCK会将信号阻塞,这样只有我们把关键的代码执行完毕之后,再去触发信号处理函数以保证数据和程序逻辑的完整性。

05

PHP如何优雅的处理信号

经常见到身边的程序员们,每当需要重启PHP-FPM进程的时候,使用的招数是kill掉所有PHP进程,然后新启动。一般情况没啥问题,但有些时候可能某个进程的任务还没执行完,直接把人家中断了略显粗暴。其实只要你给PHP的Master进程发送一条USR2信号,它便会再处理完所有任务后,重启子进程,这才是所谓的优雅~

上图是我简单写的一个例子,如果我们想让进程优雅退出的时候,只需要发送SIGTERM信号即可。需要注意的是SIGKILL和SIGSTOP信号会略过信号阻塞会将进程直接停止,还有就是信号会中断睡眠(SLEEP),sleep如果没执行完会返回剩下的秒数,有兴趣可以试试。

信号相关的知识点其实有很多,还需要继续深入研究~上文中的PHP源码为7.1.25版本,各个版本可能不太一样,如果觉得学到了点啥,顺手点个好看

责任编辑:张燕妮 来源: 假装是个程序员
相关推荐

2022-08-08 15:45:44

JavaPromise前端

2022-08-08 10:45:49

PromiseJava脚手架

2010-12-30 10:04:49

Linux入门

2012-11-23 13:14:52

2016-01-05 10:30:59

后端程序员缓存原理

2015-08-14 13:51:22

程序员

2014-10-17 18:31:05

PHP

2015-08-03 09:33:21

PH程序员一生

2015-07-13 11:32:09

PHP程序员正确姿势

2017-03-02 09:37:36

PHP程序员成长规划

2016-12-19 16:55:10

PHP程序员运维

2009-03-20 10:06:21

程序员PHP职场

2020-07-13 08:08:05

程序员软考原理

2015-06-11 13:12:56

2015-09-16 09:57:41

swoolePHP程序员

2009-01-08 09:04:18

2011-07-07 14:47:15

PHP

2013-08-20 09:33:59

程序员

2009-11-19 10:21:02

2014-08-13 10:38:01

点赞
收藏

51CTO技术栈公众号