首頁 >後端開發 >php教程 >php如何處理進程訊號(附實例)

php如何處理進程訊號(附實例)

不言
不言轉載
2018-09-28 16:05:302596瀏覽

這篇文章帶給大家的內容是關於php如何處理進程訊號(附實例),有一定的參考價值,有需要的朋友可以參考一下,希望對你有幫助。

php有一組行程控制函式PCNTL,使得php能在*nix系統中實作跟c一樣的建立子行程、使用exec函式執行程式、處理訊號等功能。

注意:pcntl這個擴充只在cli/cgi模式下可用。 mod_php和php-fpm中不可以使用。在web server環境中不要使用這組函數,因為會導致不可預料的結果。另,windows作為非類unix系統,沒有這些函數。

PCNTL 使用ticks來作為訊號處理機制(signal handle callback mechanism),可以最小程度地降低處理非同步事件時的負載。何謂ticks? Tick 是一個在程式碼段中解釋器每執行 N 個低階語句就會發生的事件,這個程式碼段需要透過declare來指定。

下面是一個隔5秒發送一個SIGALRM訊號,並由signal_handler函數獲取,然後列印一個「Caught SIGALRM」的例子:

<?php   
        declare(ticks = 1);   
      
        function signal_handler($signal) {   
            print "Caught SIGALRM/n";   
            pcntl_alarm(5);   
        }   
      
        pcntl_signal(SIGALRM, "signal_handler", true);   
        pcntl_alarm(5);   
      
        while(true){

        }
      
    ?>

其實官方的pcntl_signal效能極差,主要是PHP的函數無法直接註冊到作業系統訊號設定中,所以pcntl訊號需要依賴tick機制來完成。 pcntl_signal的實作原理是,觸發訊號後先將訊號加入一個佇列。然後在PHP的ticks回呼函數中不斷檢查是否有訊號,如果有訊號就執行PHP中指定的回呼函數,如果沒有則跳出函數。 ticks=1表示每執行1行PHP程式碼就回呼此函數。實際上大部分時間都沒有訊號產生,但ticks的函數一直會執行。 比較好的做法是去掉ticks,改用pcntl_signal_dispatch,在程式碼循環中自行處理訊號。

pcntl_signal_dispatch的實作

<?php
// 定义一个处理器,接收到SIGINT信号后只输出一行信息
function signalHandler($signo) {
    switch ($signo) {
        case SIGUSR1: echo "SIGUSR1\n"; break;
        case SIGUSR2: echo "SIGUSR2\n"; break;
        default:      echo "unknow";    break;
    }
}

//安装信号触发器器
pcntl_signal(SIGINT, &#39;signalHandler&#39;);
while (true) {
    sleep(1);
    posix_kill(posix_getpid(), SIGUSR1);///向当前进程发送SIGUSR1信号
    pcntl_signal_dispatch(); //接收到信号时,调用注册的signalHandler()
}

PCNTL的函數:

訊號處理

int pcntl_alarm ( int $seconds )

設定一個$seconds秒後發送SIGALRM訊號的計數器

bool pcntl_signal(int $signo ,callback $handler [,bool $restart_syscalls=true])

為$signo設定一個處理該訊號的回呼函數

第一個參數是訊號編號 第二個參數是訊號發生時回呼的PHP函數。 第三個參數是是否restart,是否重新註冊此訊號。這個參數如果為false,那麼此訊號只註冊處理一次。

注意:每次對 pcntl_alarm()的呼叫都會取消先前設定的alarm訊號和sleep()函數。

下面是一個隔5秒發送一個SIGALRM訊號,並由signal_handler函數獲取,然後列印一個「Caught SIGALRM」的例子:

<?php   
        declare(ticks = 1);   
      
        function signal_handler($signal) {   
            print "Caught SIGALRM/n";   
            pcntl_alarm(5);   
        }   
      
        pcntl_signal(SIGALRM, "signal_handler", true);   
        pcntl_alarm(5);   
      
        for(;;) {   
        }   
      
    ?>

#執行程式

void pcntl_exec ( string $path [, array $args [, array $envs ]] )

在目前的進程空間中執行指定程序,類似於c中的exec族函數。所謂目前空間,即載入指定程式的程式碼會覆蓋掉目前程序的空間,執行完該程式程序即結束。

<?php   
    $dir = &#39;/home/test/&#39;;   
    $cmd = &#39;ls&#39;;   
    $option = &#39;-l&#39;;   
    $pathtobin = &#39;/bin/ls&#39;;   
      
    $arg = array($cmd, $option, $dir);   
      
    pcntl_exec($pathtobin, $arg);   
    echo &#39;123&#39;;    //不会执行到该行   
    ?>

創建進程

int pcntl_fork ( void ) 为当前进程创建一个子进程
int pcntl_wait ( int &$status [, int $options ] ) 阻塞当前进程,只到当前进程的一个子进程退出或者收到一个结束当前进程的信号。 
int pcntl_waitpid ( int $pid , int &$status [, int $options ] ) 功能同pcntl_wait,区别为waitpid为等待指定pid的子进程。当pid为-1时pcntl_waitpid与pcntl_wait一样。

在pcntl_wait和pcntl_waitpid兩個函數中的$status中存了子進程的狀態信息,這個參數可以用於pcntl_wifexited、pcntl_wifstopped 、pcntl_wifsignaled、pcntl_wexitstatus、 pcntl_wtermsig、pcntl_wstopsig、pcntl_waitpid這些函數。

舉例

<?php   
    $pid = pcntl_fork();   
    if($pid) { 
        pcntl_wait($status);   
        $id = getmypid();   
        echo "parent process,pid {$id}, child pid {$pid}/n";   
    } else {   
        $id = getmypid();   
        echo "child process,pid {$id}/n";   
        sleep(2);   
    }   
    ?>

子進程在輸出child process等字樣之後sleep了2秒才結束,而父進程阻塞著直到子進程退出之後才繼續運行。

進程優先權

int pcntl_getpriority ([ int $pid [, int $process_identifier ]] ) 取得进程的优先级,即nice值,默认为0。不同的系统类型以及内核版本下 优先级可能不同(手册中为-20到20)
bool pcntl_setpriority ( int $priority [, int $pid [, int $process_identifier ]] ) 设置进程的优先级

以上是php如何處理進程訊號(附實例)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

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