pcntl_alarm定时闹钟信号详解

haotian

目的:主要测试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函数调用就太频繁,会对系统造成压力

8682 7 0
7个评论

haotian

@walkor有没有更好的办法

  • 暂无评论
walkor

信号有个作用,会打断系统调用,让系统调用立刻返回,比如read、poll等都是系统调用。
sleep也是系统调用,所以即使是sleep(100),当信号发生时sleep会立刻返回,实际不会sleep 100秒。
while循环里利用系统调用等待信号是一个很常用的做法,例如主进程监控子进程退出事件可以用wait系统调用,下一句紧接着是pcntl_signal_dispatch,这样既可以监听到子进程退出,也可以监听到其它信号并及时处理,workerman主进程就是这么处理的。

  • 暂无评论
haotian

哪有些系统调用必须要执行完,不能被信号突然打断,我们应该如何处理啊?

  • 暂无评论
haotian

还有个问题请教大牛一下 在文档说明里面
time_interval
多长时间执行一次,单位秒,支持小数,可以精确到0.001,即精确到毫秒级别。
但是在实际的里面是pcntl_alarm(1)每一秒触发一次时钟,这种0.001是怎么办到的?

  • 暂无评论
walkor

如果没装libevent扩展和event扩展则使用stream_select 的超时机制,
如果装了libevent扩展和event扩展就直接使用它们提供的定时器

  • 暂无评论
walkor

如果预见有信号打断问题要做重试的

  • 暂无评论
walkor

业务要做重试

  • 暂无评论
年代过于久远,无法发表评论

haotian

1955
积分
0
获赞数
0
粉丝数
2017-02-10 加入
🔝