PHP多进程之pcntl扩展的使用详解
安裝
1.編譯php源碼時加入
--enable-pcntl2.在現有的環境下新增加擴展
cd php-version/ext/pcntl phpize ./configure && make && make install echo "extension=pcntl.so" >> /etc/php.ini3.Mac下用brew安裝 例如我本地是php7.1.5
brew install php71-pcntl使用
header('content-type:text/html;charset=utf-8' ); // 必須加載擴展 if (!function_exists("pcntl_fork")) { die("pcntl extention is must !"); } pcntl_signal(SIGCHLD, SIG_IGN); //如果父進程不關心子進程什么時候結束,子進程結束后,內核會回收。 $pid_dir = __dir__."/pid_files"; for($i=0; $i<3; $i++){$pid = pcntl_fork(); //創建子進程 if($pid == -1){//錯誤處理:創建子進程失敗時返回-1.var_dump("fork failed");}if(!$pid){//子進程得到的$pid為0, 所以這里是子進程執行的邏輯。 //子進程代碼$pid = posix_getpid();$ppid = posix_getppid();$r = rand(0,100); //隨機數touch("$pid_dir/fork_child_process_{$i}_{$ppid}_{$pid}_{$r}");exit;}else{//父進程會得到子進程號,所以這里是父進程執行的邏輯 //如果不需要阻塞進程,而又想得到子進程的退出狀態,則可以注釋掉pcntl_wait($status)語句,或寫成: pcntl_wait($status,WNOHANG); //等待子進程中斷,防止子進程成為僵尸進程。} } $pid = posix_getpid(); $ppid = posix_getppid(); $r = rand(0,100); //隨機數 touch("$pid_dir/fork_process_pid_{$ppid}_{$pid}_$r");相關參數
pcntl_waitpid
等待或返回fork的子進程狀態。
多進程的主進程創建了子進程,那主進程如何確認子進程的狀態呢。 假如主進程需要根據子進程的狀態做不同的處理呢, 這里的狀態包括子進程被kill掉,或變成僵尸進程等。 pcntl_waitpid就可以獲取子進程的狀態碼, 通過這個狀態碼, 就可知道子進程處于什么狀態
他的用法:
返回的值可以是-1,0或者 >0的值, 如果是-1, 表示子進程出錯, 如果>0表示子進程已經退出且值是退出的子進程pid,至于如何退出, 可以通過$status狀態碼反應。 那什么時候返回0呢, 只有在option 參數為 WNOHANG且子進程正在運行時0, 也就是說當設置了options=WNOHANG時, 如果子進程還沒有退出, 此時pcntl_waitpid就會返回0
另外, 如果不設置這個參數為WNOHANG, pcntl_waitpid 就會阻塞運行, 直到子進程退出, 至于option的另外一個值WUNTRACED, 暫未理解, 不表
那么如何根據$status(狀態碼)判斷進程是如何退出呢, 如下(參數都是$status)
pcntl_wifexited
這個函數可以根據$status 判斷進程是否正常退出, 何為正常退出, 比如exit
pcntl_wexitstatus
這個函數僅在pcntl_wifexited 返回True(即正常退出)時有效, 且返回子進程退出的返回狀態碼, 這個返回狀態碼可以通過exit($s)的參數($s必須為整數時)定義
pcntl_wifsignaled
檢查子進程狀態碼是否代表由于某個信號而中斷, 比如是不是我們給他發送了term, int 等信號了
pcntl_wexitstatus
假如是發送信號而導致子進程中斷, 那么這個信號是什么信號呢, 這個函數就是獲取這個信號的
pcntl_wifstopped
僅當option選項為WUNTRACED時有效, 未理解, 不表
pcntl_wtermsig
同上
綜合實例代碼:
$res = pcntl_waitpid($pid, $status, WNOHANG); //FileLog::log("pid is $pid; wait result is $res"); if($res == -1 || $res > 0){if(!pcntl_wifexited($status)){//進程非正常退出FileLog::log("service stop unusally; pid is $pid");}else{//獲取進程終端的退出狀態碼;$code = pcntl_wexitstatus($status);FileLog::log("service stop code: $code;pid is $pid ");}if(pcntl_wifsignaled($status)){//不是通過接受信號中斷FileLog::log("service stop not by signal;pid is $pid ");}else{$signal = pcntl_wtermsig($status);FileLog::log("service stop by signal $signal;pid is $pid");} }上面的這個代碼就通過根據pcntl_waitpid的返回結果和狀態碼對子進程因為不同原因中斷做了不同的處理
來源:https://www.lvtao.net/dev/php-pcntl-process.html
總結
以上是生活随笔為你收集整理的PHP多进程之pcntl扩展的使用详解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 玫瑰之约电动车H1怎么激活4档?
- 下一篇: predis如何实现phpredis的p