注意: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,在程式碼循環中自行處理訊號。
<?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, 'signalHandler'); while (true) { sleep(1); posix_kill(posix_getpid(), SIGUSR1);///向当前进程发送SIGUSR1信号 pcntl_signal_dispatch(); //接收到信号时,调用注册的signalHandler() }
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 = '/home/test/'; $cmd = 'ls'; $option = '-l'; $pathtobin = '/bin/ls'; $arg = array($cmd, $option, $dir); pcntl_exec($pathtobin, $arg); echo '123'; //不会执行到该行 ?>
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中文網其他相關文章!