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_setprioritypcntl_setpriority ( int $priority [ , int $pid = getmypid() [, int $process_identifier = PRIO_PROCESS]])設定程序的優先權● getmypid() 取得目前php程序的pid##● 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}次\r\n"; exit(); } }
以上是關於php多進程程式設計(程式碼詳解)的詳細內容。更多資訊請關注PHP中文網其他相關文章!