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

2020-07-16 10:06:10

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

以上就是關於php多進程程式設計(程式碼詳解)的詳細內容,更多請關注TW511.COM其它相關文章!