内容来自http://baike.baidu.com/view/6339376.htm?fr=aladdin
int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue);
which有三种,即setitimer支持三种定时器类型
ITIMER_REAL: 以系统真实的时间来计算,它送出SIGALRM信号。
ITIMER_VIRTUAL: -以该进程在用户态下花费的时间来计算,它送出SIGVTALRM信号。
ITIMER_PROF: 以该进程在用户态下和内核态下所费的时间来计算,它送出SIGPROF信号。
接着看itimerval这个结构体:
struct itimerval { struct timeval it_interval; struct timeval it_value; }; struct timeval { long tv_sec; long tv_usec; };
it_value指定初始定时时间
如果只指定it_value,就是实现一次定时
如果同时指定 it_interval,则超时后,系统会重新初始化it_value为it_interval,实现重复定时;
两者都清零,则会清除定时器
tv_sec提供秒级精度,tv_usec提供微秒级精度,以值大的为先,注意1s = 1000000us。
ovalue用来保存先前的值,常设为NULL。
上代码:
#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <signal.h> #include <time.h> #include <sys/time.h> void sigroutine(int signo) { switch (signo) { case SIGALRM: printf("Catch a signal -- SIGALRM \n"); //signal(SIGALRM, sigroutine); break; } fflush(stdout);//清除写缓冲区 return; } int main() { struct itimerval value, ovalue; printf("process id is %d\n", getpid()); signal(SIGALRM, sigroutine); value.it_value.tv_sec = 1; value.it_value.tv_usec = 0; value.it_interval.tv_sec = 1; value.it_interval.tv_usec = 0; setitimer(ITIMER_REAL, &value, &ovalue); for(;;); }
process id is 23393
Catch a signal -- SIGALRM
Catch a signal -- SIGALRM
其中用到了signal函数,简单介绍一下
void (*signal(int signum, void(* handler)(int) ) ) (int);
下面的情况可以产生Signal:
1、 按下CTRL+C产生SIGINT
2、硬件中断,如除0,非法内存访问(SIGSEV)等等
3、Kill函数可以对进程发送Signal
4、Kill命令。实际上是对Kill函数的一个包装
5、软件中断。如当Alarm Clock超时(SIGURG),当Reader中止之后又向管道写数据(SIGPIPE),等等