使用alarm配合信号实现sleep

author: selfimpr

blog: http://blog.csdn.net/lgg201

mail: lgg860911@yahoo.com.cn


APUE中描述Solaris 9是使用alarm实现的sleep, 其语义如下:

如果在sleep之前有一个未到期的alarm时钟, 则中断时钟

1. 如果剩余时间大于sleep时间: 执行sleep, 在sleep完了之后, 继续执行剩余的时间(前面的alarm被中断时的剩余时间 减去 sleep时间)

2. 如果剩余时间小于sleep时间: 执行sleep, 但是sleep的是剩余时间, 并由sleep返回执行后的剩余时间(也就是alarm被中断时的剩余时间 减去 实际sleep时间)


下面是一个简化的实现, 语义为:

如果sleep时发现已经有一个alarm时钟, 中断alarm时钟, sleep指定时间, 然后, 继续未完成的alarm时钟(两个时间不互相影响)

这个语义比较鸡肋...忽忽...


#include <apue.h> #include <signal.h> static int ud_sleep(int); static void sig_alrm(int); static void ud_wait(int); void main(void) { printf("first test case: \n"); printf("begin at: %d\n", time(NULL)); //设置一个时钟 alarm(6); printf("after alarm(6) at: %d\n", time(NULL)); //模拟等待3秒 ud_wait(3); printf("after sleep(3) at: %d\n", time(NULL)); //进行一次5秒的睡眠 ud_sleep(5); printf("after ud_sleep(5) at: %d\n", time(NULL)); //等一会儿, 看看第一个alarm剩余的时间到了会发生什么 ud_wait(5); } /** * 睡眠seconds秒 * 与信号的交互规则: 如果之前有alarm时钟, 则以将seconds时间插入到原来的alarm中方式处理 */ static int ud_sleep(int seconds) { sigset_t newmask, oldmask, susmask; int unslept1, unslept2; //设置SIGALRM的处理器 if ( signal(SIGALRM, sig_alrm) == SIG_ERR ) err_sys("signal(SIGALRM) error"); //阻塞SIGALRM信号 sigemptyset(&newmask); sigaddset(&newmask, SIGALRM); if ( sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0 ) err_sys("sigprocmask(SIG_BLOCK) error"); //设置alarm时钟 unslept1 = alarm(0); alarm(seconds); //以原有的信号掩码等待alarm时钟 susmask = oldmask; sigdelset(&susmask, SIGALRM); sigsuspend(&susmask); //读取剩余睡眠时间并重置信号处理器和信号掩码 unslept2 = alarm(0); if ( signal(SIGALRM, sig_alrm) == SIG_ERR ) err_sys("signal(SIGALRM) error"); if ( sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0 ) err_sys("sigprocmask(SIG_SETMASK) error"); alarm(unslept1); return unslept2; } static void sig_alrm(int signo) { signal(SIGALRM, sig_alrm); printf("receive SIGALRM at: %d\n", time(NULL)); } static void ud_wait(int imax) { int i = 0, j; while ( i ++ < imax * 10000 ) { j = 0; while ( j ++ < 30000) { ; } } }


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值