当前位置: 移动技术网 > IT编程>开发语言>PHP > PHP进程通信基础之信号

PHP进程通信基础之信号

2017年12月12日  | 移动技术网IT编程  | 我要评论

使用信号通信。可以使用kill -l 来查看当前系统的信号类型。
每个信号所代表的的详细含义,请查看我的这篇文章:
使用信号的时候可以通过php --version 来查看当前php的版本。已决定使用哪种方式来进行进程间的信号通信。

[root@roverliang ipc]# php --version

php 5.6.24 (cli) (built: aug 15 2016 19:14:02)
copyright (c) 1997-2016 the php group
zend engine v2.6.0, copyright (c) 1998-2016 zend technologies

使用pcntl_signal_dispatch 函数 需要php 版本(php 5 >= 5.3.0, php 7)

如果php版本小于5.3.一些大公司可能会低于这个版本。这个时候会使用 declare(ticks=1),意思为每执行一条低级指令,
就会去检测是否出现该信号。详细的介绍可以查看

官网解释如下: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 信号
 */
function signalhandal($signal)
{
  switch ($signal) {
    case sigint:
      //do something
      break;
    case sighup:
      //do something
      break;
    default :
      //do something
      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 "parent progress pid:{$parentpid}\n";

//定义一个信号处理函数

function sighandler($signal) {
  if ($signal == sigint) {
    $pid = getmypid();
    exit("{$pid} process, killed!".php_eol);
  }
}

//php version < 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 "i am child: ".getmypid(). " and i am running !".php_eol;
      sleep(rand(1,3));
    }
  } elseif($pid > 0) {
    $child_list[] = $pid;
  } else {
    die('fork fail!'.php_eol);
  }
}

sleep(5);
foreach ($child_list as $key => $pid) {
  posix_kill($pid, sigint);
}

sleep(2);
echo "{$parentpid} parent is end".php_eol;

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网