php pcntl_fork程序不死掉的解決辦法:1、開啟相應的PHP程式碼檔案;2、檢視php多執行緒處理程式碼;3、通過「public function installSignal(){pcntl_signal(SIGTERM, [$this, 'sigHandle'])...};」方式安裝訊號處理器即可。
php入門到就業線上直播課:進入學習
Apipost = Postman + Swagger + Mock + Jmeter 超好用的API偵錯工具:
本教學操作環境:windows7系統、PHP8.1版、Dell G3電腦。
php pcntl_fork 程序不死掉怎麼辦?
問題描述
第一次使用php多執行緒處理任務引起殭屍程序問題,原因是子程序沒有傳送結束訊號,父程序沒有等待子程序的結束
理解
pcntl_fork 返回-1:開啟程序失敗 0:表示當前是子程序 大於零:當前是父程序,返回值是子程序pid
fork的子程序程式碼執行完後必須exit(),傳送結束訊號。否則繼續fork子程序。獲取父程序pid = posix_getppid(), 當前pid = getmypid()或者posix_getpid()
父程序最好等待子程序執行結束回收子程序通過pcntl_wait()、pcntl_waitpid()實現同步、非同步等待父程序迴圈子程序:
$result = pcntl_waitpid($pid_, $status, WNOHANG);
$result等於-1時代表子程序結束
登入後複製
訊號處理器的作用是根據處理子程序返回的結束訊號進行處理
public function run()
{
$this->installSignal();
// 獲取當前pid getmypid()
$current_pid = posix_getpid();
// dump('當前pid:'.$current_pid);
cli_set_process_title('設定process_title');
$pids = [];
for($i = 0; $i < $this->forks; $i ++){
$pid = pcntl_fork();
if($pid == -1){
die('程序開啟失敗');
}else if($pid){
// 父程序得到子程序pid
dump('父程序');
dump('父程序得到子程序pid'.$pid);
$pids[] = $pid;
}else{
dump('子程序');
dump('父程序pid-'.posix_getppid());
// 子程序
dump("第{$i}個程序");
sleep(10);
dump("第{$i}個程序:等待10s");
// 傳送結束訊號
// exit(-1);
// posix_kill(getmypid(), SIGKILL);
posix_kill(getmypid(), SIGHUP);
}
}
// 父程序等待子程序執行結束
while($pids){
foreach($pids as $k => $pid_){
dump('父程序不阻塞-等待子程序結束');
$result = pcntl_waitpid($pid_, $status, WNOHANG);
dump($result);
if($result === -1){
dump('子程序返回狀態'.$status);
unset($pids[$k]);
}
}
}
/**
* 訊號處理器
*/
public function sigHandle($signo)
{
switch ($signo) {
case SIGTERM:
// 處理SIGTERM訊號
dump('處理SIGTERM訊號');
exit;
break;
case SIGHUP:
//處理SIGHUP訊號
dump('處理SIGHUP訊號');
exit(SIGHUP);
break;
case SIGUSR1:
dump('處理SIGUSR1訊號');
break;
default:
// 處理所有其他訊號
}
}
/**
* 安裝訊號處理器
*
*/
public function installSignal()
{
pcntl_signal(SIGTERM, [$this, 'sigHandle']);
pcntl_signal(SIGHUP, [$this,'sigHandle']);
pcntl_signal(SIGUSR1, [$this,'sigHandle']);
}
登入後複製
推薦學習:《》
以上就是php pcntl_fork 程序不死掉怎麼辦的詳細內容,更多請關注TW511.COM其它相關文章!