PHP进程退出信号_一文吃透 PHP 进程信号处理

背景

前两周老大给安排了一个任务,写一个监听信号的包。因为我司的项目是运行在容器里边的,每次上线,需要重新打包镜像,然后启动。在重新打包之前,Dokcer会先给容器发送一个信号,然后等待一段超时时间(默认10s)后,再发送SIGKILL信号来终止容器

现在有一种情况,容器中有一个常驻进程,该常驻进程的任务是不断的消费队列里的消息。假设现在要上线,需要关杀掉容器,Docker给容器里跑的常驻进程发送一个信号,告诉它我10s后会将你关闭,假设现在已经过了9秒,常驻进程刚从队列中取出一条消息,1s内还没将后续逻辑执行完,进程就已经被杀了,此时这条消息就丢失了,且可能会产生脏数据

上边就是这次任务的背景,需要通过监听信号来决定后续如何操作。对于上边这种情况,当常驻进程收到Docker发送的关闭信号时,将该进程阻塞即可,一直sleep,直到杀掉容器。OK,清楚背景之后,下边就介绍一下PHP中的信号(后边会再整理一篇这个包如何写,并将包发布到https://packagist.org/,供需要的小伙伴使用)

一、在Linux操作系统中有哪些信号

1、简单介绍信号

信号是事件发生时对进程的通知机制,有时又称为软件中断。一个进程可以向另一个进程发送信号,比如子进程结束时都会向父进程发送一个SIGCHLD(17号信号)来通知父进程,所以有时信号也被当作一种进程间通信的机制。

在linux系统下,通常我们使用 kill -9 XXPID来结束一个进程,其实这个命令的实质就是向某进程发送SIGKILL(9号信号),对于在前台进程我们通常用Ctrl+c快捷键来结束运行,该快捷键的实质是向当前进程发送SIGINT(2号信号),而进程收到该信号的默认行为是结束运行

2、常用信号

下边这些信号,可以使用kill -l命令进行查看

576ae95f1c1ec9ece41cf90e3c99a1bc.png

下边介绍几个比较重要且常用的信号:

信号名

信号值

信号类型

信号描述

SIGHUP

1

终止进程(终端线路挂断)

本信号在用户终端连接(正常或非正常、结束时发出, 通常是在终端的控制进程结束时, 通知同一session内的各个作业, 这时它们与控制终端不再关联

SIGQUIT

2

终止进程(中断进程)

程序终止(interrupt、信号, 在用户键入INTR字符(通常是Ctrl-C、时发出

SIGQUIT

3

建立CORE文件终止进程,并且生成CORE文件

进程,并且生成CORE文件SIGQUIT 和SIGINT类似, 但由QUIT字符(通常是Ctrl-、来控制. 进程在因收到SIGQUIT退出时会产生core文件, 在这个意义上类似于一个程序错误信 号

SIGFPE

8

建立CORE文件(浮点异常)

SIGFPE 在发生致命的算术运算错误时发出. 不仅包括浮点运算错误, 还包括溢 出及除数为0等其它所有的算术的错误

SIGKILL

9

终止进程(杀死进程)

SIGKILL 用来立即结束程序的运行. 本信号不能被阻塞, 处理和忽略

SIGSEGV

11

SIGSEGV 试图访问未分配给自己的内存, 或试图往没有写权限的内存地址写数据

SIGALRM

14

终止进程(计时器到时)

SIGALRM 时钟定时信号, 计算的是实际的时间或时钟时间. alarm函数使用该信号

SIGTERM

15

终止进程(软件终止信号)

SIGTERM 程序结束(terminate、信号, 与SIGKILL不同的是该信号可以被阻塞和处理. 通常用来要求程序自己正常退出. shell命令kill缺省产生这个信号

SIGCHLD

17

忽略信号(当子进程停止或退出时通知父进程)

SIGCHLD 子进程结束时, 父进程会收到这个信号

SIGVTALRM

26

终止进程(虚拟计时器到时)

SIGVTALRM 虚拟时钟信号. 类似于SIGALRM, 但是计算的是该进程占用的CPU时间

SIGIO

29

忽略信号(描述符上可以进行I/O)

SIGIO 文件描述符准备就绪, 可以开始进行输入/输出操作

二、PHP中处理信号相关函数

PHP的pcntl扩展以及posix扩展为我们提供了若干操作信号的方法(若想使用这些函数,需要先安装这几个扩展)

下边具体介绍几个我在本次任务中用到的方法:

declare

declare结构用来设定一段代码的执行指令。declare的语法和其它流程控制结构相似

declare (directive)

statement

复制代码

directive部分允许设定declare代码段的行为。目前只认识两个指令:ticks和encoding。declare代码段中的 statement部分将被执行——怎样执行以及执行中有什么副作用出现取决于directive中设定的指令

Ticks

Tick(时钟周期)是一个在declare代码段中解释器每执行N条可计时的低级语句就会发生的事件N的值是在declare 中的directive部分用ticks=N来指定的。不是所有语句都可计时。通常条件表达式和参数表达式都不可计时。在每个tick中出现的事件是由register_tick_function()来指定的,注意每个 tick 中可以出现多个事件

更详细的内容,可查看官方文档:https://www.php.net/manual/zh/control-structures.declare.php

declare(ticks=1);//每执行一条时,触发register_tick_function()注册的函数

$a=1;//在注册之前,不算

function test(){//定义一个函数

echo "执行\n";

}

register_tick_function('test');//该条注册函数会被当成低级语句被执行

for($i=0;$i<=2;$i++){//for算一条低级语句

$i=$i;//赋值算一条

}

输出:六个“执行”

复制代码

pcntl_signal

pcntl_signal,安装一个信号处理器

pcntl_signal ( int $signo , callback $handler [, bool $restart_syscalls = true ] ) : bool

复制代码

函数pcntl_signal()为signo指定的信号安装一个新的信号处理器

declare(ticks = 1);

pcntl_signal(SIGINT,function(){

echo "你按了Ctrl+C".PHP_EOL;

});

while(1){

sleep(1);//死循环运行低级语句

}

输出:当按Ctrl+C之后,会输出“你按了Ctrl+C”

复制代码

posix_kill

posix_kill,向进程发送一个信号

posix_kill ( int $pid , int $sig ) : bool

复制代码

第一个参数为进程ID,第二个参数为你要发送的信号

a.php

declare(ticks = 1);

echo getmypid();//获取当前进程id

pcntl_signal(SIGINT,function(){

echo "你给我发了SIGINT信号";

});

while(1){

sleep(1);

}

b.php

posix_kill(执行1.php时输出的进程id, SIGINT);

复制代码

pcntl_signal_dispatch

pcntl_signal_dispatch,调用等待信号的处理器

pcntl_signal_dispatch ( void ) : bool

复制代码

函数pcntl_signal_dispatch()调用每个等待信号通过pcntl_signal()安装的处理器

echo "安装信号处理器...\n";

pcntl_signal(SIGHUP,  function($signo) {

echo "信号处理器被调用\n";

});

echo "为自己生成SIGHUP信号...\n";

posix_kill(posix_getpid(), SIGHUP);

echo "分发...\n";

pcntl_signal_dispatch();

echo "完成\n";

?>

输出:

安装信号处理器...

为自己生成SIGHUP信号...

分发...

信号处理器被调用

完成

复制代码

pcntl_async_signals()

异步信号处理,用于启用无需 ticks (这会带来很多额外的开销)的异步信号处理。(PHP>=7.1)

pcntl_async_signals(true); // turn on async signals

pcntl_signal(SIGHUP,  function($sig) {

echo "SIGHUP\n";

});

posix_kill(posix_getpid(), SIGHUP);

输出:

SIGHUP

复制代码

三、PHP中处理信号量的方式

前边我们知道我们可以通过declare(ticks=1)和pcntl_signal组合的方式监听信号,即每一条PHP低级语句,就会检查一次当前进程是否有未处理的信号,这其实是十分耗性能的。

pcntl_signal的实现原理是,触发信号后先将信号加入一个队列中。然后在PHP的ticks回调函数中不断检查是否有信号,如果有信号就执行PHP中指定的回调函数,如果没有则跳出函数。

PHP_MINIT_FUNCTION(pcntl)

{

php_register_signal_constants(INIT_FUNC_ARGS_PASSTHRU);

php_pcntl_register_errno_constants(INIT_FUNC_ARGS_PASSTHRU);

php_add_tick_function(pcntl_signal_dispatch TSRMLS_CC);

return SUCCESS;

}

复制代码

在PHP5.3之后,有了pcntl_signal_dispatch函数。这个时候将不在需要declare,只需要在循环中增加该函数,就可以调用信号通过了:

echo getmypid();//获取当前进程id

pcntl_signal(SIGUSR1,function(){

echo "触发信号用户自定义信号1";

});

while(1){

pcntl_signal_dispatch();

sleep(1);//死循环运行低级语句

}

复制代码

大家都知道PHP的ticks=1表示每执行1行PHP代码就回调此函数。实际上大部分时间都没有信号产生,但ticks的函数一直会执行。如果一个服务器程序1秒中接收1000次请求,平均每个请求要执行1000行PHP代码。那么PHP的pcntl_signal,就带来了额外的 1000 * 1000,也就是100万次空的函数调用。这样会浪费大量的CPU资源。比较好的做法是去掉ticks,转而使用pcntl_signal_dispatch,在代码循环中自行处理信号。

pcntl_signal_dispatch 函数的实现:

void pcntl_signal_dispatch()

{

//.... 这里略去一部分代码,queue即是信号队列

while (queue) {

if ((handle = zend_hash_index_find(&PCNTL_G(php_signal_table), queue->signo)) != NULL) {

ZVAL_NULL(&retval);

ZVAL_LONG(&param, queue->signo);

/* Call php signal handler - Note that we do not report errors, and we ignore the return value */

/* FIXME: this is probably broken when multiple signals are handled in this while loop (retval) */

call_user_function(EG(function_table), NULL, handle, &retval, 1, &param TSRMLS_CC);

zval_ptr_dtor(&param);

zval_ptr_dtor(&retval);

}

next = queue->next;

queue->next = PCNTL_G(spares);

PCNTL_G(spares) = queue;

queue = next;

}

}

复制代码

但是上边这种,也有个恶心的地方就是,它得放在死循环中。PHP7.1之后出来了一个完成异步的信号接收并处理的函数:

pcntl_async_signals

//a.php

echo getmypid();

pcntl_async_signals(true);//开启异步监听信号

pcntl_signal(SIGUSR1,function(){

echo "触发信号";

posix_kill(getmypid(),SIGSTOP);

});

posix_kill(getmypid(),SIGSTOP);//给进程发送暂停信号

//b.php

posix_kill(文件1进程, SIGCONT);//给进程发送继续信号

posix_kill(文件1进程, SIGUSR1);//给进程发送user1信号

复制代码

通过pcntl_async_signals方法,就不用再写死循环了。

监听信号的包:

https://github.com/Rain-Life/monitorSignal

复制代码

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/289777.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

GitHub Copilot 现已登陆 Visual Studio!

激动人心的好消息来了&#xff0c;GitHub 在3月29日发布博客&#xff0c;宣布 Github Copilot 现在可以在 Visual Studio 中使用。我们知道 Visual Studio 的 IntelliCode 本身已经很智能了, 现在又迎来了 Copilot, 编程体验将进入新的篇章。如何安装? 首先&#xff0c;您…

经典实用SQL语句大全汇总

目 录 1.随机取3条记录 2.随机选记录 3.删除重复记录 4.创建数据库 5.列出表里的所有的列名 6.选择从10到15的记录 7.压缩数据库

java之写接口回调编程经验改进

1、问题 在一个类里面数据的变化需要在另外一个类里面动态得到&#xff0c;比如在我的异步任务里面下载的数据&#xff0c;需要在UI界面的ProgressDialog里面动态显示&#xff0c;我们需要在异步任务里面写个接口&#xff0c;然后接口里面有一些函数&#xff0c;至于在Progres…

Mac Ubuntu ----端口被占用

Mac下使用lsof&#xff08;list open files&#xff09;来查看端口占用情况&#xff0c;lsof 是一个列出当前系统打开文件的工具。 使用 lsof 会列举所有占用的端口列表&#xff1a; 1$ lsof使用less可以用于分页展示&#xff0c;如&#xff1a; 1$ lsof | less也可以使用 -i 查…

iOS 9音频应用播放音频之音量设置与声道设置

iOS 9音频应用播放音频之音量设置与声道设置 iOS 9音频应用音量设置 音量又称响度、音强&#xff0c;是指人耳对所听到的声音大小强弱的主观感受&#xff0c;其客观评价尺度是声音的振幅大小。在iOS 9音频应用的应用中&#xff0c;经常会出现播放的音乐音量过大或者过小。此时i…

nginx配置文件中的location中文详解

location 语法:location [|~|~*|^~] /uri/ { … }默认:否 上下文:server 这个指令随URL不同而接受不同的结构。你可以配置使用常规字符串和正则表达式。如果使用正则表达式&#xff0c;你必须使用 ~* 前缀选择不区分大小写的匹配或者 ~ 选择区分大小写的匹配。 确定 哪个locati…

php fpm工作原理,什么是phpfpm的工作原理?

什么是phpfpm的工作原理&#xff1f;发布时间&#xff1a;2020-07-13 15:12:53来源&#xff1a;亿速云阅读&#xff1a;181作者&#xff1a;Leah什么是phpfpm的工作原理&#xff1f;针对这个问题&#xff0c;这篇文章详细介绍了相对应的分析和解答&#xff0c;希望可以帮助更多…

C#对象映射器之Mapster

简介Mapster是一个快&#xff0c;小巧&#xff0c;功能强大的对象映射.Net框架例子我有两个Model类且他们的属性一致&#xff0c;我们将 SourceObjectTest赋值给DestObjectTest该怎么做&#xff1f;SourceObjectTest sourceObject new SourceObjectTest(); sourceObject.Name …

如何关闭Struts2的webconsole.html

出于安全目的&#xff0c;在禁用了devMode之后&#xff0c;仍然不希望其他人员看到webconsole.html页面&#xff0c;则可以直接删除webconsole.html 的源文件&#xff0c; 它的位置存在于&#xff1a; 我们手工删除 struts2-core-*.jar\org\apache\struts2\interceptor\debuggi…

UIView 的基础

UIView•什么是控件&#xff1f;-屏幕上的所有UI元素都叫做控件&#xff0c;也有人叫做视图、组件-按钮&#xff08;UIButton&#xff09;、文本&#xff08;UILabel&#xff09;都是控件•控件的共同属性有哪些&#xff1f;-尺寸-位置-背景色-......-•苹果将控件的共同属性都…

记录平时编程或者阅读英文文档的时候不认识的英文单词

一、目的 英语虽然过了4级&#xff0c;但是还是很多英文单词不认识&#xff0c;为了以后能更好的阅读英文文档和函数的英文注释&#xff0c;记录自己不认识的英文单词&#xff0c;方便以后复习&#xff0c;这里后面会持续更新&#xff0c;因为放在网上这个平台不会弄丢。 二、…

php本地的调试安装,教你本地安装、运行、调试PHP程序

安装工具对学习PHP的新手来说&#xff0c;WINDOWS下环境配置是一件很困难的事&#xff1b;至少对于我来说本地调试PHP程序比登天还要困难&#xff0c;繁荣拖沓的各种程序。以前我博客程序用的是PJBLOG&#xff0c;本地的IIS就完全可以搞定&#xff0c;但是自从换了WordPress之后…

如何从一个 C# 的 dump 中挖到机器相关的信息?

前段时间有位朋友问我&#xff0c;如何从 dump 中提取出哪些和机器相关的信息&#xff1f;比如&#xff1a;机器内存大小&#xff0c;cpu核数&#xff0c;机器名&#xff0c;机器的环境变量 等等。那如何提取到里面的信息呢&#xff1f;当然我也没说全部可以提取的到。。。这里…

9.3磁盘及文件系统管理详解

练习&#xff1a;4 调整其预留百分比为3%#tune2fs -m 3 -L DATA /dev/sda75 以重新挂载此文件系统为不更新访问时间戳&#xff0c;并验证其效果#stat /backup/inittab# cat /backup/inittab# stat# mount -o remount,noatime /backup# cat # stat6 对此文件系统强行做一次检测e…

LeetCode() Word Search II

超时&#xff0c;用了tire也不行&#xff0c;需要再改。 class Solution {class TrieNode { public:// Initialize your data structure here.TrieNode() {for(int i0;i<26;i)next[i]NULL;isString false;}TrieNode *next[26];bool isString; };class Trie { public:Trie()…

java之通过FileChannel实现文件复制

1、FileChanel介绍 Java NIO FileChannel是连接文件的通道&#xff0c;从文件中读取数据和将数据写入文件。Java NIO FileChannel类是NIO用于替代使用标准Java IO API读取文件的方法。 FileInputStream的getChannel方法获取的文件通道是只读的&#xff0c;当然通过FileOutput…

Python版的Hello World

print “Hello World” 或者print("Hello World")

如何获取当前C#程序所有线程的调用栈信息 ?

咨询区 Daniel Sperry请问如何获取 .NET 程序当前所有线程的调用栈信息&#xff1f;我知道在 java 中只需调用 java.lang.Thread.getAllStackTraces() 方法即可。回答区 Will Calderwood在 .NET 中并不容易实现&#xff0c;但可以使用诊断库 ClrMD &#xff0c;可以在 nuget 上…

JS

为什么80%的码农都做不了架构师&#xff1f;>>> function getQueryString(name) {var reg new RegExp("(^|&)" name "([^&]*)(&|$)"),r window.location.search.substr(1).match(reg);if(r ! null) {return unescape(r[2]); }r…