PHP进程通信基础之信号
使用信号通信。可以使用kill-l来查看当前系统的信号类型。
每个信号所代表的的详细含义,请查看我的这篇文章:https://www.nhooo.com/article/106040.htm
使用信号的时候可以通过php--version来查看当前PHP的版本。已决定使用哪种方式来进行进程间的信号通信。
[root@roverliangipc]#php--version PHP5.6.24(cli)(built:Aug15201619:14:02) Copyright(c)1997-2016ThePHPGroup ZendEnginev2.6.0,Copyright(c)1998-2016ZendTechnologies
使用pcntl_signal_dispatch函数需要PHP版本(PHP5>=5.3.0,PHP7)
如果PHP版本小于5.3.一些大公司可能会低于这个版本。这个时候会使用declare(ticks=1),意思为每执行一条低级指令,
就会去检测是否出现该信号。详细的介绍可以查看https://www.nhooo.com/article/48340.htm
官网解释如下:Tick(时钟周期)是一个在declare代码段中解释器每执行N条可计时的低级语句就会发生的事件。N的值是在declare中的directive部分用ticks=N来指定的。
那么什么是低级语句呢:如下代码所示:
for($i=0;$i<3;$i++){
echo$i.PHP_EOL;
}
那么这个for循环中就含有三条低级指令。每输出一条$i。就会去检测下是否发生了已注册的事件,可想而知,这样效率是比较低的。所以如果检测到自己的PHP大于等于5.3。就使用pcntl_singal_dispath来进行信号派送。
主进程在启动的时候注册一些信号处理函数。
/**
*@param$signal信号
*/
functionsignalHandal($signal)
{
switch($signal){
caseSIGINT:
//dosomething
break;
caseSIGHUP:
//dosomething
break;
default:
//dosomething
break;
}
}
然后将信号处理器与信号处理函数绑定:
//根据不同的信号,安装不同的信号处理器 pcntl_signal(SIGINT,'signalHandal'); pcntl_signal(SIGHUP,'signalHandal'); pcntl_signal(SIGUSR1,'signalHandla');
在子进程监听信号,如果出现该信号,就调用预安装的信号处理函数
//分配信号。 pcntl_signal_dispatch($signal);
我们来整理下思路:
1、定义信号发生所需要处理事件的函数
2、将信号和信号处理函数绑定,称为信号安装。
3、信号监听或者分发,出现信号调用已安装的信号。
理解好上面的信号概念,我们来看一个demo:
<?php
$parentpid=posix_getpid();
echo"parentprogresspid:{$parentpid}\n";
//定义一个信号处理函数
functionsighandler($signal){
if($signal==SIGINT){
$pid=getmypid();
exit("{$pid}process,Killed!".PHP_EOL);
}
}
//phpversion<5.3.每执行一条低级指令,就检查一次是否出现该信号。效率损耗很大。
//declare(ticks=1);
$child_list=[];
//注册一个信号处理器。当发出该信号的时候对调用已定义的函数
pcntl_signal(SIGINT,'sighandler');
for($i=0;$i<3;$i++){
$pid=pcntl_fork();
if($pid==0){
//子进程
while(true){
//调用已安装的信号信号处理器,为了检测是否有新的信号等待dispatching
pcntl_signal_dispatch();
echo"Iamchild:".getmypid()."andiamrunning!".PHP_EOL;
sleep(rand(1,3));
}
}elseif($pid>0){
$child_list[]=$pid;
}else{
die('forkfail!'.PHP_EOL);
}
}
sleep(5);
foreach($child_listas$key=>$pid){
posix_kill($pid,SIGINT);
}
sleep(2);
echo"{$parentpid}parentisend".PHP_EOL;