目的:主要测试pcntl_alarm的功能
测试代码一
<?php
declare(ticks = 1);
function signal_handler($signal) {
print "catch you ";
pcntl_alarm(5);
}
pcntl_signal(SIGALRM, "signal_handler", true);
pcntl_alarm(5);
while(1) {
}
?>
正确代码,证实每次pcntl_alarm预定一个计时器,然后当计时器到时间的时候,就会给当前进程触发SIGALRM信号
测试代码二
<?php
declare(ticks = 1);
function signal_handler($signal) {
print "catch you ";
// pcntl_alarm(5);
}
pcntl_signal(SIGALRM, "signal_handler", true);
pcntl_alarm(5);
while(1) {
}
?>
注释掉触发函数中的信号计时器,我们发现信号只触发了一次,这就说明pcntl_alarm只会一次触发
测试代码三
<?php
declare(ticks = 1);
function signal_handler($signal) {
print "catch you ";
// pcntl_alarm(5);
}
pcntl_signal(SIGALRM, "signal_handler", true);
pcntl_alarm(5);
sleep(100);
将while循环换成sleep,发现在5秒的时候信号触发,程序结束,这是为什么呢,对应手册中提到的注意点,pcntl_alarm信号会提前结束掉sleep()函数
测试代码四
之前发过文章说过declare和pcntl_signal_dispatch两个函数的区别,那么这里用后面一个函数实现呢?
<?php
function signal_handler($signal) {
print "Caught SIGALRM ";
pcntl_alarm(5);
}
pcntl_signal(SIGALRM, "signal_handler", true);
pcntl_alarm(5);
for(;;) {
sleep(100);
pcntl_signal_dispatch();
}
?>
这里我用到了一个特性 第一个特性就是sleep和SIGALRM信号之间的关系
其实你看可以去掉sleep,但是这样dispatch函数调用就太频繁,会对系统造成压力
@walkor有没有更好的办法
信号有个作用,会打断系统调用,让系统调用立刻返回,比如read、poll等都是系统调用。
sleep也是系统调用,所以即使是sleep(100),当信号发生时sleep会立刻返回,实际不会sleep 100秒。
while循环里利用系统调用等待信号是一个很常用的做法,例如主进程监控子进程退出事件可以用wait系统调用,下一句紧接着是pcntl_signal_dispatch,这样既可以监听到子进程退出,也可以监听到其它信号并及时处理,workerman主进程就是这么处理的。
哪有些系统调用必须要执行完,不能被信号突然打断,我们应该如何处理啊?
还有个问题请教大牛一下 在文档说明里面
time_interval
多长时间执行一次,单位秒,支持小数,可以精确到0.001,即精确到毫秒级别。
但是在实际的里面是pcntl_alarm(1)每一秒触发一次时钟,这种0.001是怎么办到的?
如果没装libevent扩展和event扩展则使用stream_select 的超时机制,
如果装了libevent扩展和event扩展就直接使用它们提供的定时器
如果预见有信号打断问题要做重试的
业务要做重试