查看: 1684|回复: 0

[PHP学习] PHP进程通信基础之信号

发表于 2017-12-13 08:00:02

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

  1. [root@roverliang ipc]# php --version
  2. PHP 5.6.24 (cli) (built: Aug 15 2016 19:14:02)
  3. Copyright (c) 1997-2016 The PHP Group
  4. 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),意思为每执行一条低级指令,
就会去检测是否出现该信号。详细的介绍可以查看 http://www.jb51.net/article/48340.htm

官网解释如下:Tick(时钟周期)是一个在 declare 代码段中解释器每执行 N 条可计时的低级语句就会发生的事件。N 的值是在 declare 中的 directive 部分用 ticks=N 来指定的。

那么什么是低级语句呢:如下代码所示:

  1. for ($i = 0; $i < 3; $i++) {
  2. echo $i.PHP_EOL;
  3. }
复制代码

那么这个for 循环中就含有三条低级指令。每输出一条$i。就会去检测下是否发生了已注册的事件,可想而知,这样效率是比较低的。所以如果检测到自己的PHP大于等于5.3 。就使用pcntl_singal_dispath 来进行信号派送。

主进程在启动的时候注册一些信号处理函数。

  1. /**
  2. * @param $signal 信号
  3. */
  4. function signalHandal($signal)
  5. {
  6. switch ($signal) {
  7. case SIGINT:
  8. //do something
  9. break;
  10. case SIGHUP:
  11. //do something
  12. break;
  13. default :
  14. //do something
  15. break;
  16. }
  17. }
复制代码

然后将信号处理器与信号处理函数绑定:

  1. //根据不同的信号,安装不同的信号处理器
  2. pcntl_signal(SIGINT, 'signalHandal');
  3. pcntl_signal(SIGHUP, 'signalHandal');
  4. pcntl_signal(SIGUSR1, 'signalHandla');
复制代码

在子进程监听信号,如果出现该信号,就调用预安装的信号处理函数

  1. //分配信号。
  2. pcntl_signal_dispatch($signal);
复制代码

我们来整理下思路:
1、定义信号发生所需要处理事件的函数
2、将信号和信号处理函数绑定,称为信号安装。
3、信号监听或者分发,出现信号调用已安装的信号。

理解好上面的信号概念,我们来看一个demo:

  1. <?php
  2. $parentpid = posix_getpid();
  3. echo "parent progress pid:{$parentpid}\n";
  4. //定义一个信号处理函数
  5. function sighandler($signal) {
  6. if ($signal == SIGINT) {
  7. $pid = getmypid();
  8. exit("{$pid} process, Killed!".PHP_EOL);
  9. }
  10. }
  11. //php version < 5.3 .每执行一条低级指令,就检查一次是否出现该信号。效率损耗很大。
  12. //declare(ticks=1);
  13. $child_list = [];
  14. //注册一个信号处理器。当发出该信号的时候对调用已定义的函数
  15. pcntl_signal(SIGINT, 'sighandler');
  16. for($i = 0; $i < 3; $i++) {
  17. $pid = pcntl_fork();
  18. if ($pid == 0) {
  19. //子进程
  20. while (true) {
  21. //调用已安装的信号信号处理器,为了检测是否有新的信号等待dispatching
  22. pcntl_signal_dispatch();
  23. echo "I am child: ".getmypid(). " and i am running !".PHP_EOL;
  24. sleep(rand(1,3));
  25. }
  26. } elseif($pid > 0) {
  27. $child_list[] = $pid;
  28. } else {
  29. die('fork fail!'.PHP_EOL);
  30. }
  31. }
  32. sleep(5);
  33. foreach ($child_list as $key => $pid) {
  34. posix_kill($pid, SIGINT);
  35. }
  36. sleep(2);
  37. echo "{$parentpid} parent is end".PHP_EOL;
复制代码



回复

使用道具 举报