• 首页
  • vue
  • TypeScript
  • JavaScript
  • scss
  • css3
  • html5
  • php
  • MySQL
  • redis
  • jQuery
  • pcntl_waitpid()

    (PHP 4 >= 4.1.0, PHP 5, PHP 7)

    等待或返回fork的子进程状态

    说明

    pcntl_waitpid(int $pid,int &$status[,int $options= 0]): int

    挂起当前进程的执行直到参数$pid指定的进程号的进程退出,或接收到一个信号要求中断当前进程或调用一个信号处理函数。

    如果$pid指定的子进程在此函数调用时已经退出(俗称僵尸进程),此函数将立刻返回。关于waitpid更详细的规范请参见您系统的waitpid(2)手册。

    参数

    $pid

    参数$pid的值可以是以下之一:

    $pid可选值
    <-1等待任意进程组ID等于参数$pid给定值的绝对值的进程。
    -1等待任意子进程;与pcntl_wait函数行为一致。
    0等待任意与调用进程组ID相同的子进程。
    > 0等待进程号等于参数$pid值的子进程。
    Note:

    指定-1作为$pid的值等同于pcntl_wait()提供(负的$options)。

    $status

    pcntl_waitpid()将会存储状态信息到$status参数上,这个通过$status参数返回的状态信息可以用以下函数pcntl_wifexited(),pcntl_wifstopped(),pcntl_wifsignaled(),pcntl_wexitstatus(),pcntl_wtermsig()以及pcntl_wstopsig()获取其具体的值。

    $options

    如果您的操作系统(多数BSD类系统)允许使用wait3,您可以提供可选的$options参数。如果这个参数没有提供,wait将会被用作系统调用。如果wait3不可用,提供参数$options不会有任何效果。$options的值可以是0 或者以下两个常量或两个常量“或运算”结果(即两个常量代表意义都有效)。

    $options可用的值
    WNOHANG如果没有子进程退出立刻返回。
    WUNTRACED子进程已经退出并且其状态未报告时返回。

    返回值

    pcntl_waitpid()返回退出的子进程进程号,发生错误时返回-1,如果提供了WNOHANG作为option(wait3可用的系统)并且没有可用子进程时返回0。

    参见

    • pcntl_fork()在当前进程当前位置产生分支(子进程)。译注:fork是创建了一个子进程,父进程和子进程都从fork的位置开始向下继续执行,不同的是父进程执行过程中,得到的fork返回值为子进程号,而子进程得到的是0。
    • pcntl_signal()安装一个信号处理器
    • pcntl_wifexited()检查状态代码是否代表一个正常的退出。
    • pcntl_wifstopped()检查子进程当前是否已经停止
    • pcntl_wifsignaled()检查子进程状态码是否代表由于某个信号而中断
    • pcntl_wexitstatus()返回一个中断的子进程的返回代码
    • pcntl_wtermsig()返回导致子进程中断的信号
    • pcntl_wstopsig()返回导致子进程停止的信号
    <?php
    $childs = array();
    // Fork some process.
    for($i = 0; $i < 10; $i++) {
      $pid = pcntl_fork();
      if($pid == -1)
        die('Could not fork');
      if ($pid) {
        echo "parent \n";
        $childs[] = $pid;
      } else {
        // Sleep $i+1 (s). The child process can get this parameters($i).
        sleep($i+1);
        
        // The child process needed to end the loop.
        exit();
      }
    }
    while(count($childs) > 0) {
      foreach($childs as $key => $pid) {
        $res = pcntl_waitpid($pid, $status, WNOHANG);
        
        // If the process has already exited
        if($res == -1 || $res > 0)
          unset($childs[$key]);
      }
      
      sleep(1);
    }
    ?>
    
    please note, if you using configure option --enable-sigchild(Enable PHP's own SIGCHLD handler) when complie php(under linux 2.6.18-53.1.13.el5.centos.plus and php 5.2.5 as I know), pcntl_waitpid and pcntl_wait in php script would never return the child pid, because the build in handle get it first.
    <?php
    declare(ticks = 1);
    function zp_handler($signal) {
      $id = pcntl_waitpid(-1, $status, WNOHANG);
      if (pcntl_wifexited($status)) 
      {
        printf("Removed Chlid id: %d \n",$id);
        printf("Chlid status: %d \n",pcntl_wexitstatus($status));
      }
    }
    //pcntl_signal_dispatch();
    pcntl_signal(SIGCHLD, "zp_handler");
    //pcntl_signal_dispatch();
    //
    $pid = pcntl_fork();
    if ($pid == 0)
    {
      print "#1 Hi, I'm child process".PHP_EOL;
      sleep(3);
      return 10;
    }
    else 
    {
      print "#1parent process id:".$pid.PHP_EOL;
      $pid = pcntl_fork();
      if ($pid == 0)
      {  print "#2 Hi, I'm child process".PHP_EOL;
        sleep(10);
        exit(20);
      } 
      else 
      {
        print "#2parent process id:".$pid.PHP_EOL;
        for ($i=0; $i <10 ; $i++) { 
          print "wait..".PHP_EOL;
          sleep(10);
        }
      }
    }
    ?>
    

    上篇:pcntl_wait()

    下篇:pcntl_wexitstatus()