使用时需要引入的头文件:
#include <sys/time.h>
setitimer函数原型:
int setitimer(int which, const struct itimerval *new_value,
struct itimerval *old_value);
其中which参数表示类型,可选的值有:
ITIMER_REAL: 以系统真实的时间来计算,它送出SIGALRM信号。
ITIMER_VIRTUAL:以该进程在用户态下花费的时间来计算,它送出SIGVTALRM信号。
ITIMER_PROF: 以该进程在用户态下和内核态下所费的时间来计算,它送出SIGPROF信号。
紧接着的new_value和old_value均为itimerval结构体,先看一下itimerval结构体定义:
struct itimerval {
struct timeval it_interval; /* next value */
struct timeval it_value; /* current value */
};
struct timeval {
time_t tv_sec; /* seconds */
suseconds_t tv_usec; /* microseconds */
};
itimeval又是由两个timeval结构体组成,timeval包含tv_sec和tv_usec两部分,其中tv_se为秒,tv_usec为微秒(即1/1000000秒)
其中的new_value参数用来对计时器进行设置,it_interval为计时间隔,it_value为延时时长。
settimer工作机制是,先对it_value倒计时,当it_value为零时触发信号,然后重置为it_interval,继续对it_value倒计时,一直这样循环下去。
基于此机制,setitimer既可以用来延时执行,也可定时执行。
注意:假如it_value为0是不会触发信号的,所以要能触发信号,it_value得大于0;如果it_interval为零,只会延时,不会定时(也就是说只会触发一次信号)。
old_value参数,通常用不上,设置为NULL,它是用来存储上一次setitimer调用时设置的new_value值。
下面例子中表示的是在setitimer方法调用成功后,延时1微秒便触发一次SIGALRM信号,以后每隔1秒触发一次SIGALRM信号。
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <sys/time.h>
void signalHandler(int signo)
{
switch (signo) {
case SIGALRM: //定时信号(由系统定时事件触发)
printf("Caught the SIGALRM signal!\n");
break;
case SIGINT: //终端按下CTRL + C键触发
printf("Caught the SIGINT signal!\n");
exit(0);
break;
}
}
int main(int argc, char *argv[])
{
/* 安装信号处理函数 */
signal(SIGALRM, signalHandler);
signal(SIGINT, signalHandler);
struct itimerval new_value, old_value;
new_value.it_value.tv_sec = 0;
new_value.it_value.tv_usec = 1;
new_value.it_interval.tv_sec = 1;
new_value.it_interval.tv_usec = 0;
setitimer(ITIMER_REAL, &new_value, &old_value);
while (1) {
}
return 0;
}
此外,在应用过程中还需要注意:
setitimer一个进程中只能有一个 下一个会覆盖前一个的定时 想一个进程多个定时器只能自己实现。
setitimer() 不支持在同一进程中同时使用多次以支持多个定时器。
linux有关定时器(setitimer)的叙述是这样的:
linux系统给每个进程提供了3个定时器,每个定时器在各自不同的域里面计数。当任何一个timer计数到结束了,系统就发送一个信号(signal)给该进程,同时计数器重置。