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();//子进程执行完后应该退出,不然会继续执行后面的逻辑 }
● pcntl_wait(int &$status[, int $options = 0]):等待或返回fork的子进程状态,相当于pcntl_waitpid(-1,int &$status[,int $options = 0])
● pcntl_waitpid(int $pid , int &$status[,int $options = 0]) $status是作为一下函数的参数
● pcntl_wifexited(int $status) 检查子进程状态代码是否代表正常退出,
● pcntl_wexistatus(int $status) 返回一个中断的子进程返回代码,仅在正常中断才有效
● pcntl_wifsignaled(int $status) 检查子进程是否由某个未捕获的信号退出的。是返回true,否返回false
● 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) }
● pcntl_alarm(int $seconds):为进程设置一个alarn闹钟信号
● pcntl_signal(int $signo, callback $handler [, bool $restart_syscalls = true ] )为指定的信号安装一个新的信号处理器
● 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(); };
● pcntl_getpriority( int $pid = getmypid() [, int $process_identifier = PRIO_PROCESS ])获取进程的优先级
● pcntl_setpriority( int $priority [ , int $pid = getmypid() [, int $process_identifier = PRIO_PROCESS]])设置进程的优先级
● getmypid() 获取当前php进程的pid
● 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}次rn"; exit(); } }