首页 > PHP技术 > php高级 > php多进程编程详解
2020
04-09

php多进程编程详解



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();
     }
 }

扫码芷若 获取免费视频学习资料

编程学习

查 看2019高级编程视频教程免费获取