Linux环境编程之信号处理(三、利用alarm()和pause()函数实现sleep()函数)

        sleep()是执行挂起一段时间,而alarm()函数是定时器,pause()函数则是挂起进程,当出现信号打断时,才会继续往前执行;

        先来分享下alarm()函数,alarm()函数用来设置一个定时器,当时间超时时,会产生SIGALRM信号,该信号默认是终止该进程;


        #include

        unsigned int alarm(unsigned int seconds);

        参数:unsigned int seconds表示经过seconds秒后会产生信号SIGALRM;

        返回值:0或者以前设置的闹钟时间的余留秒数;

      

        1、每个进程只能有一个闹钟时钟;

        2、调用alarm()时,假如以前调用过alarm()函数,并且指定的时间还没到(即:还没超时,没有发送SIGALRM信号)。那么此次调用的alarm()函数会替代掉上一次的alarm()函数,并且返回值为上次调用alarm()函数中没有用完的时间;

        3、本次调用alarm(0)函数会取消上一次未超时的定时器,返回剩余的时间;


        下面也看看pause()函数:

        #include

        int pause(void);

        pause()会把进程挂起来,直到一个信号处理程序执行完后,才会继续运行;

        

        最后看看sleep()函数:

        #include

        unsigned int sleep(unsigned int seconds);

        该函数使调用进程被挂起了,直到seconds后(超时后),或者捕捉到一个信号并且从信号处理函数返回后,该进程才能继续往下执行;返回值和alarm()函数的一样。

        

        下面是利用alarm()函数和pause()函数来实现下sleep()函数:

#include
#include
#include
#include
 
static jmp_buf envAlarm;
 
static void myAlarm(int signo)// 自定义的SIGALRM定时器信号的处理函数
 {
     longjmp(envAlarm, 1);// 局部跳转
 }
 
 unsigned int mySleep(unsigned int sec)
 {
     unsigned int minSec, ret, tmp = 0;
 
     if (signal(SIGALRM, myAlarm) == SIG_ERR)// 绑定SIGALRM和myAlarm()函数
         return sec;
 
     minSec = alarm(0);// 得到上一次未超时的定时器所剩的时间,防止冲刷掉上一次的真正的定时器(本函数的是实现sleep())
     ret = sec > minSec ? (sec-minSec):0;// 得到返回值
 
     if (setjmp(envAlarm) == 0){ // 局部跳转
         if (minSec) // 这里考虑的是如果前面有设置定时器
             alarm(minSec < sec ? minSec:sec);// 利用时间短的去做定时器
         else // 如果前面没有设置定时器,则直接利用参数去挂起进程
             alarm(sec);
 
         sleep(9);// 这里是测试上一个alarm()函数到pause()之间延迟导致SIGALRM信号处理完后,pause()还没有触犯,则进程一直挂起
         pause();
     }   
     tmp = alarm(0);// 计算在睡眠时 有可能被其他信号打断,而剩下一些未睡眠的时间
     if (minSec > sec){// 如果前面登记的定时器比睡眠时间长,则退出后还要把定时器复位为剩下时间的定时器
         alarm(minSec-sec);// 定时器还剩多少时间,重新设置定时器
     } 
    if (tmp)ret += tmp; // 统计所有剩余时间
 
     return ret;
 }
 
 int main(int argc, char* argv[])
 {
     unsigned int otherSec;
     alarm(6);
     //sleep(4);
     otherSec = mySleep(3);
     printf("mySleep resturn:%u\n", otherSec);
     return 0;
 }

        上面是通过局部跳转来实现sleep()函数的,当然还有使用信号屏蔽字来实现的,后面有时间会实现下。不知道上面的代码是否有漏洞,虽然这个代码量不多,但是要考虑的问题还是比较多的。如果有朋友发现了漏洞,欢迎指正!!谢谢!!

        

        转载请注明作者和原文出处,原文地址: http://blog.csdn.net/yuzhihui_no1/article/details/44855419
        若有不正确之处,望大家指正,共同学习!谢谢!!!


你可能感兴趣的:(面试,unix环境高级编程)