欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

php 中的信号处理操作实例详解

程序员文章站 2022-09-06 23:42:49
本文实例讲述了php 中的信号处理操作。分享给大家供大家参考,具体如下:首先我们需要了解几个函数pcntl_signal 安装信号处理器,也就是当指定信号发生时,调用函数。pcntl_alarm 指定...

本文实例讲述了php 中的信号处理操作。分享给大家供大家参考,具体如下:

首先我们需要了解几个函数

pcntl_signal   安装信号处理器,也就是当指定信号发生时,调用函数。
pcntl_alarm   指定秒数后向进程发送sigalrm信号。
posix_getpid 返回当前进程id
posix_kill       给指定进程发送信号

一些常用的信号说明

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

sigint
程序终止(interrupt)信号, 在用户键入intr字符(通常是ctrl-c)时发出。

sigquit
和sigint类似,但由quit字符(通常是ctrl-/)来控制;进程在因收到sigquit退出时会产生core文件,
在这个意义上类似于一个程序错误信号。

sigkill
用来立即结束程序的运行。本信号不能被阻塞、处理和忽略。如果管理员发现某个进程终止不了,可尝试发送这个信号。

sigterm
程序结束(terminate)信号, 与sigkill不同的是该信号可以被阻塞和处理。通常用来要求程序自己正常退出,
shell命令kill缺省产生这个信号。如果进程终止不了,我们才会尝试sigkill。

sigusr1
留给用户使用

sigusr2
留给用户使用

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

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

例1:

<?php
declare(ticks = 1);
//信号处理函数
function sig_func() {
  echo "child exit \r\n";
}
//设置信号处理器
pcntl_signal(sigchld, 'sig_func');
$pid = pcntl_fork();
if($pid == -1) {
  die('fork error');
} else if ($pid) {
  pcntl_wait($status);
} else {
  echo "child... \r\n";
  exit;
}

当子进程退出时,会向父进程发送sigchld信号,我们通过设置信号处理器,成功的处理信号。

例2:

<?php
declare(ticks = 1);
//信号处理函数
function sig_func($signo) {
  switch($signo) {
    case sigchld: {
      echo "child sigchld \r\n";
      break;
    }
    case sigterm: {
      echo "child sigterm \r\n";
      break;
    }
    default:
      //处理所有其他信号
      break;
  }
}
//设置信号处理器
pcntl_signal(sigchld, 'sig_func');
//设置信号处理器
pcntl_signal(sigterm, 'sig_func');
$pid = pcntl_fork();
if($pid == -1) {
  die('fork error');
} else if ($pid) {
  pcntl_wait($status);
} else {
  sleep(3);
  echo "child \r\n";
  sleep(3);
  posix_kill(getmypid(), sigterm);
  exit;
}

父进程等待子进程的退出,子进程等待3秒后输出child,再等待3秒后向自身发送结束程序信号。

例3:

<?php
declare(ticks = 1);
//信号处理函数
function sig_func($signo) {
  switch($signo) {
    case sigchld: {
      echo "child sigchld \r\n";
      break;
    }
    /*这里要把处理sigterm信号的代码注释掉
    case sigterm: {
      echo "child sigterm \r\n";
      break;
    }*/
    default:
      //处理所有其他信号
      break;
  }
}
//设置信号处理器
pcntl_signal(sigchld, 'sig_func');
//设置信号处理器,也注释掉
//不然当父进程发向子进程发送sigterm信号时,子进程不会退出,还会继续执行
//我们的信号处理函数把sigterm给忽略了
//pcntl_signal(sigterm, 'sig_func');
$pid = pcntl_fork();
if($pid == -1) {
  die('fork error');
} else if ($pid) {
  sleep(30);
  posix_kill($pid, sigterm);
} else {
  $cnt = 0;
  for(;;) {
    sleep(3);
    echo $cnt, '-';
    ++$cnt;
  }
  exit;
}

父进程在等待30秒后,向子进程发送sigterm结束程序信号。如果我们设置了sigterm信号的处理器,并且在自定义信号处理器中并没有杀死该进程,则该子进程会一直运行下去。

pcntl_signal()函数仅仅是注册信号和它的处理方法,真正接收到信号并调用其处理方法的是pcntl_signal_dispatch()函数。

例4:

<?php
//使用ticks需要php 4.3.0以上版本
//declare(ticks = 1);
function sig_func() {
  echo "sigalrm \r\n";
}
//设置信号处理器
pcntl_signal(sigalrm, 'sig_func');
pcntl_alarm(3);

通过函数pcntl_alarm()3秒后给进程发送sigalrm信号,但信号处理函数并未调用。
原因是我们注释了declare(ticks = 1);这段代码,而又没有调用pcntl_signal_dispatch()函数。

declare(ticks = 1);表示每执行一条低级指令,就检查一次信号,如果检测到注册的信号,就调用其信号处理器。但是这种处理方式效率很低,建议在代码循环中通过pcntl_signal_dispatch()来处理信号。

<?php
//使用ticks需要php 4.3.0以上版本
//declare(ticks = 1);
function sig_func() {
  echo "sigalrm \r\n";
}
//设置信号处理器
pcntl_signal(sigalrm, 'sig_func');
pcntl_alarm(3);
//因为3秒后pcntl_alarm函数才会给进程发送sigalrm信号
//所以我们通过sleep函数等待3秒后,调用pcntl_signal_dispatch()来处理信号
sleep(3);
pcntl_signal_dispatch();

pcntl_signal_dispatch()这个函数是php5.3以上才支持的,如果你的php版本大于5.3,建议使用这个方法调用信号处理器。
5.3以下的版本需要在注册信号之前加上:declare(ticks = 1);