PHP前端开发

## 使用 pcntl_async_signals 和 pcntl_wait 时,信号回调函数为何无法触发?

百变鹏仔 1个月前 (12-15) #PHP
文章标签 回调

pcntl_async_signals 和 pcntl_wait 的协同使用

问题背景

在 php 中,pcntl_async_signals(true) 设置异步信号处理,允许信号回调函数在信号到达时被调用,而 pcntl_wait($status) 通常用于在父进程中阻塞等待子进程终止。

问题

使用 pcntl_async_signals(true) 启用异步信号处理后,发现 pcntl_wait($status) 阻塞父进程时无法触发信号回调函数的执行。

原因

在使用 pcntl_wait($status) 阻塞父进程时,pcntl_signal 的第三个参数默认为 true。这意味着信号到达时不会中断父进程的执行,也不会调用信号回调函数。

解决方法

有两种方法可以解决此问题:

  1. 使用循环阻塞父进程

使用循环(如 while (!$this->shouldstop) { usleep(1); })阻塞父进程,而不是使用 pcntl_wait($status)。这种方法允许 usleep() 触发时钟滴答(tick),从而提供信号回调能力。

  1. 在 pcntl_signal 中禁用中断

将 pcntl_signal 的第三个参数设置为 false。这将允许信号中断父进程的执行并调用信号回调函数。

示例

以下示例修改了 pcntl_signal 的调用,以禁用中断:

pcntl_signal(SIGTERM, [$this, 'masterTermHandler'], false);

更改后,pcntl_wait($status) 将阻塞父进程,并且当 sigterm 信号到达时,mastertermhandler 信号回调函数仍将被调用。