首頁 >後端開發 >php教程 >關於php多進程程式設計(程式碼詳解)

關於php多進程程式設計(程式碼詳解)

藏色散人
藏色散人轉載
2020-01-31 17:03:533044瀏覽

關於php多進程程式設計(程式碼詳解)

php多進程程式設計

#PHP的進程控制支援實作了Unix方式的進程建立, 程式執行,信號處理以及進程的中斷。進程控制不能被應用在Web伺服器環境,當其被用於Web服務環境時可能會帶來意外的結果。

 

pcntl函數

pcntl_fork():在目前行程目前位置產生分支(子行程)。

譯註:fork是創建了一個子進程,父進程和子進程都從fork的位置開始向下繼續執行,不同的是父進程執行過程中,得到的fork返回值為子進程號,而子行程得到的是0

<?php
$pid = pcntl_fork();
//父进程和子进程都会执行下面代码
if ($pid == -1) {
    //错误处理:创建子进程失败时返回-1.
     die(&#39;could not fork&#39;);
} 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 (&#39;fork error on Task object&#39;);
 }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教學

! ###

以上是關於php多進程程式設計(程式碼詳解)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:cnblogs.com。如有侵權,請聯絡admin@php.cn刪除