page contents

php多进程编程详解

PHP的进程控制支持实现了Unix方式的进程创建, 程序执行, 信号处理以及进程的中断。 进程控制不能被应用在Web服务器环境,当其被用于Web服务环境时可能会带来意外的结果。

attachments-2020-04-PbMjW4Gr5e8e8acd9a7f0.jpg

PHP的进程控制支持实现了Unix方式的进程创建, 程序执行, 信号处理以及进程的中断。 进程控制不能被应用在Web服务器环境,当其被用于Web服务环境时可能会带来意外的结果。


pcntl函数


  • pcntl_fork():在当前进程当前位置产生分支(子进程)。

  • 译注:fork是创建了一个子进程,父进程和子进程 都从fork的位置开始向下继续执行,不同的是父进程执行过程中,得到的fork返回值为子进程 号,而子进程得到的是0

<?php
$pid = pcntl_fork();
//父进程和子进程都会执行下面代码
if ($pid == -1) {
    //错误处理:创建子进程失败时返回-1.
     die('could not fork');
} else if ($pid) {
     //父进程会得到子进程号,所以这里是父进程执行的逻辑
     pcntl_wait($status); //等待子进程中断,防止子进程成为僵尸进程。
} else {
     //子进程得到的$pid为0, 所以这里是子进程执行的逻辑。
    exit();//子进程执行完后应该退出,不然会继续执行后面的逻辑
}
  1. pcntl_wait(int &$status[, int $options = 0]):等待或返回fork的子进程状态,相当于pcntl_waitpid(-1,int &$status[,int $options = 0])
  2. pcntl_waitpid(int $pid , int &$status[,int $options = 0]) $status是作为一下函数的参数
  3. pcntl_wifexited(int $status) 检查子进程状态代码是否代表正常退出,
  4. pcntl_wexistatus(int $status) 返回一个中断的子进程返回代码,仅在正常中断才有效
  5. pcntl_wifsignaled(int $status) 检查子进程是否由某个未捕获的信号退出的。是返回true,否返回false
  6. pcntl_wtermsig(int $status)返回导致子进程中断的信号,当pcntl_wifsignaled返回true时有效
<?php
 echo "主进程\n";
 $pid = pcntl_fork();
 //父进程和子进程都会执行这些代码
 if($pid == -1 ){
     //创建子进程失败会返回-1
     throw new Exception ('fork error on Task object');
 }else if($pid){
     //创建成功会父进程会得到子进程的pid
     echo "等待子进程执行";
     pcntl_wait($status);//等待子进程中断
     echo "子进程执行状态:";
     echo "是否正常退出:",pcntl_wifexited($status),"\n";
     echo "子进程返回的代码:",pcntl_wexitstatus($status),"\n";//仅在pcntl_wifexited返回true时生效,只能是int,输出123
     echo "子进程是否是由于某个未捕获的信号退出的:",pcntl_wifsignaled($status),"\n";//如果是kill -9|-15 杀死的进程返回true
     echo "导致子进程中断的信号:",pcntl_wtermsig($status),"\n"; 输出 9 | 15
     var_dump($status);
 }else{
     //创建成功子进程会得到pid=0
     sleep(2);
     echo "子进程执行完毕\n";
         exit(123)
 }
  1. pcntl_alarm(int $seconds):为进程设置一个alarn闹钟信号
  2. pcntl_signal(int $signo, callback $handler [, bool $restart_syscalls = true ] )为指定的信号安装一个新的信号处理器
  3. pcntl_signal_get_handler(int $signo) 获取指定信号的处理函数
<?php
 
echo "设置3秒之后发送闹钟信号\n";
pcntl_alarm(3);
 
function dealSigalarm(){
    echo "收到信号 SIGALRM \n退出程序。。。\n";
    exit();
}
 
echo "安装信号处理器\n";
pcntl_signal(SIGALRM,"dealSigalarm");//对于不能被阻塞、处理和忽略的信号,php为这些事件注册信号处理函数会产生一个致命错误 SIGSTOP,SIGKILL
var_dump(pcntl_signal_get_handler(SIGUSR1));//输出dealSigalarm
pcntl_signal(SIGUSR1,function(){
    echo "收到用户自定义信号\n";
});
$i = 1;
while(1){
    sleep(1);
    echo $i++,"\n";
    echo "分发... \n";
    pcntl_signal_dispatch();
};
  1. pcntl_getpriority( int $pid = getmypid() [, int $process_identifier = PRIO_PROCESS ])获取进程的优先级
  2. pcntl_setpriority( int $priority [ , int $pid = getmypid() [, int $process_identifier = PRIO_PROCESS]])设置进程的优先级
  3. getmypid() 获取当前php进程的pid
  4. posix_getpid() 获取当前进程的pid
<?php
 /**
 * php进程的优先级
 */
 
 for($i = 1;$i<=5;$i++){
     $pid = pcntl_fork();
     if($pid == -1){
         throw new Exception("fork error on task object");
     }else if ($pid){
         pcntl_wait($status);
     }else{
         $end_time = time()+3;
         $k = 0;
         while(time()<=$end_time){
             $k++;
         }
         $pid = getmypid();
         echo "当前进程id:".$pid,"优先级:",pcntl_getpriority($pid);
         pcntl_setpriority($i);
         echo "修改之后的优先级为:",pcntl_getpriority(),"\n";
         echo "执行了进程{$i} {$k}次\r\n";
         exit();
     }
 }


attachments-2020-04-bj7K82W45e8e8a8d5b6f3.jpg

  • 发表于 2020-04-09 10:39
  • 阅读 ( 537 )
  • 分类:PHP开发

你可能感兴趣的文章

相关问题

0 条评论

请先 登录 后评论
Pack
Pack

1135 篇文章

作家榜 »

  1. 轩辕小不懂 2403 文章
  2. 小柒 1316 文章
  3. Pack 1135 文章
  4. Nen 576 文章
  5. 王昭君 209 文章
  6. 文双 71 文章
  7. 小威 64 文章
  8. Cara 36 文章