## 使用 pcntl_async_signals 和 pcntl_wait 时,信号回调函数为何无法触发?
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。这意味着信号到达时不会中断父进程的执行,也不会调用信号回调函数。
解决方法
有两种方法可以解决此问题:
- 使用循环阻塞父进程
使用循环(如 while (!$this->shouldstop) { usleep(1); })阻塞父进程,而不是使用 pcntl_wait($status)。这种方法允许 usleep() 触发时钟滴答(tick),从而提供信号回调能力。
- 在 pcntl_signal 中禁用中断
将 pcntl_signal 的第三个参数设置为 false。这将允许信号中断父进程的执行并调用信号回调函数。
示例
以下示例修改了 pcntl_signal 的调用,以禁用中断:
pcntl_signal(SIGTERM, [$this, 'masterTermHandler'], false);
更改后,pcntl_wait($status) 将阻塞父进程,并且当 sigterm 信号到达时,mastertermhandler 信号回调函数仍将被调用。