我的编程空间,编程开发者的网络收藏夹
学习永远不晚

PHP进程信号如何处理

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

PHP进程信号如何处理

这篇文章主要介绍“PHP进程信号如何处理”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“PHP进程信号如何处理”文章能帮助大家解决问题。

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

1、简单介绍信号

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

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

2、常用信号

下边这些信号,可以使用kill -l命令进行查看PHP进程信号如何处理

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

信号名信号值信号类型信号描述
SIGHUP1终止进程(终端线路挂断)本信号在用户终端连接(正常或非正常、结束时发出, 通常是在终端的控制进程结束时, 通知同一session内的各个作业, 这时它们与控制终端不再关联
SIGQUIT2终止进程(中断进程)程序终止(interrupt、信号, 在用户键入INTR字符(通常是Ctrl-C、时发出
SIGQUIT3建立CORE文件终止进程,并且生成CORE文件进程,并且生成CORE文件SIGQUIT 和SIGINT类似, 但由QUIT字符(通常是Ctrl-、来控制. 进程在因收到SIGQUIT退出时会产生core文件, 在这个意义上类似于一个程序错误信 号
SIGFPE8建立CORE文件(浮点异常)SIGFPE 在发生致命的算术运算错误时发出. 不仅包括浮点运算错误, 还包括溢 出及除数为0等其它所有的算术的错误
SIGKILL9终止进程(杀死进程)SIGKILL 用来立即结束程序的运行. 本信号不能被阻塞, 处理和忽略
SIGSEGV11
SIGSEGV 试图访问未分配给自己的内存, 或试图往没有写权限的内存地址写数据
SIGALRM14终止进程(计时器到时)SIGALRM 时钟定时信号, 计算的是实际的时间或时钟时间. alarm函数使用该信号
SIGTERM15终止进程(软件终止信号)SIGTERM 程序结束(terminate、信号, 与SIGKILL不同的是该信号可以被阻塞和处理. 通常用来要求程序自己正常退出. shell命令kill缺省产生这个信号
SIGCHLD17忽略信号(当子进程停止或退出时通知父进程)SIGCHLD 子进程结束时, 父进程会收到这个信号
SIGVTALRM26终止进程(虚拟计时器到时)SIGVTALRM 虚拟时钟信号. 类似于SIGALRM, 但是计算的是该进程占用的CPU时间
SIGIO29忽略信号(描述符上可以进行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 中可以出现多个事件更详细的内容。

<?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
<?php
declare(ticks = 1);
echo getmypid();//获取当前进程id
pcntl_signal(SIGINT,function(){
   echo "你给我发了SIGINT信号";
});
while(1){
   sleep(1);
}

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

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

pcntl_signal_dispatch ( void ) : bool

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

<?php
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)

<?php
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,只需要在循环中增加该函数,就可以调用信号通过了:

<?php
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_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

<?php
//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

关于“PHP进程信号如何处理”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注编程网行业资讯频道,小编每天都会为大家更新不同的知识点。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

PHP进程信号如何处理

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

PHP进程信号如何处理

这篇文章主要介绍“PHP进程信号如何处理”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“PHP进程信号如何处理”文章能帮助大家解决问题。一、在Linux操作系统中有哪些信号1、简单介绍信号信号是事件发
2023-07-05

php如何处理信号

这篇文章将为大家详细讲解有关php如何处理信号,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。我们常用的信号kill sigterm sigkill【kill命令】ctrl+c sigint【键盘发出】
2023-06-21

【linux】进程信号——信号的保存和处理

文章目录 一、阻塞信号1.1 信号的相关概念1.2 在内核中的构成 二、捕捉信号概念2.1 内核态和用户态2.2 信号捕捉流程图 三、信号操作3.1 sigset_t信号集3.2 信号集操作函数3.2.1 更改block表s
2023-08-20

golang进程信号怎么处理

在Go语言中,可以使用`os/signal`包来处理进程信号。下面是一个处理 `SIGINT` 和 `SIGTERM` 信号的示例代码:package mainimport ("fmt""os""os/signal""syscall")
2023-10-26

python中如何对信号进行处理

这篇文章主要介绍了python中如何对信号进行处理,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。什么是信号信号(signal)-- 进程间通讯的一种方式,也可作为一种软件中断
2023-06-20

Android Init进程对信号的处理流程详细介绍

Android Init进程对信号的处理流程 在Android中,当一个进程退出(exit())时,会向它的父进程发送一个SIGCHLD信号。父进程收到该信号后,会释放分配给该子进程的系统资源;并且父进程需要调用wait()或waitpi
2022-06-06

Linux exit函数与进程退出时信号的处理

在Linux中,exit()函数和信号处理是两个不同的概念,但它们都与进程的退出有关exit()函数:exit()函数是一个C库函数,用于正常终止进程。当一个程序调用exit()函数时,它会执行以下操作:关闭所有打开的文件描述符释放进程
Linux exit函数与进程退出时信号的处理
2024-08-30

PHP开发微信公众号:如何进行客户服务管理

PHP开发微信公众号:如何进行客户服务管理,需要具体代码示例引言:随着移动互联网的快速发展,微信公众号成为企业开展客户服务的重要渠道之一。众所周知,微信公众号可以通过图文、语音、视频等多种形式与用户进行交互。本文将介绍如何利用PHP开发微信
2023-10-26

如何在微信公众号上进行PHP开发

如何在微信公众号上进行PHP开发微信公众号已经成为了一个非常受欢迎的社交平台,许多企业和个人都希望能够在微信公众号上进行开发,以吸引更多的用户和粉丝。PHP作为一种广泛使用的后端编程语言,也可以应用于微信公众号的开发中。本文将介绍如何使用P
2023-10-27

如何在PHP开发中处理多线程和进程管理?

如何在PHP开发中处理多线程和进程管理?简介:在PHP开发中,多线程和进程管理是一个重要的话题。随着应用程序变得越来越复杂,处理并发和高并发访问请求的能力变得至关重要。本文将介绍如何在PHP开发中处理多线程和进程管理的技术和工具。一、多线程
如何在PHP开发中处理多线程和进程管理?
2023-11-03

怎么在python中中对信号进行处理

今天就跟大家聊聊有关怎么在python中中对信号进行处理,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。python可以做什么Python是一种编程语言,内置了许多有效的工具,Pyth
2023-06-14

windows异常进程如何处理

处理Windows异常进程的方法通常有以下几种:1. 结束进程:打开任务管理器,在“进程”选项卡中找到异常进程,右键点击选择“结束任务”。这将强制终止该进程,并关闭它的所有相关任务。2. 重启计算机:有时,异常进程可能无法通过结束任务来解决
2023-10-09

python语音信号处理详细教程

在深度学习中,语音的输入都是需要处理的,下面这篇文章主要给大家介绍了关于python语音信号处理的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2023-01-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录