内容来自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_interval指定间隔时间
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()被调用前申明,handler中为这个函数的名字。当接收到一个类型为sig的信号时,就执行handler 所指定的函数。这个函数应有如下形式的定义:
void func(int sig);
(2)SIG_IGN
这个符号表示忽略该信号,执行了相应的signal()调用后,进程会忽略类型为sig的信号。
(3)SIG_DFL
这个符号表示恢复系统对信号的默认处理。
下面的情况可以产生Signal:
1、 按下CTRL+C产生SIGINT
2、硬件中断,如除0,非法内存访问(SIGSEV)等等
3、Kill函数可以对进程发送Signal
4、Kill命令。实际上是对Kill函数的一个包装
5、软件中断。如当Alarm Clock超时(SIGURG),当Reader中止之后又向管道写数据(SIGPIPE),等等