通常需要周期性或者定时做一些事情的时候,要用到timer,以下两种常见的方式,来实现周期性功能。
一、alarm()+signal()
在精度要求不高的情况下,可以用signal。只能以秒为单位。
#include <pthread.h>
#include <stdlib.h>
#include <stdint.h>
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
#include <signal.h>
void timer_cb(int sig)
{
static cnt=0;
if(SIGALRM == sig)
{
printf("timer %d\n",cnt++);
alarm(1); //重新定时触发,达到循环目的
}
return;
}
int main(void)
{
signal(SIGALRM, timer_cb); //注册定时器信号,和功能函数
alarm(1); //触发定时器信号
getchar();
return 0;
}
二、setitimer()+signal()
要实现精度较高的定时功能的话,就要使用setitimer函数。
setitimer()为Linux的API,setitimer()有两个功能,一是指定一段时间后,才执行某个function,二是每间隔一段时间就执行某个function。
int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue));
参数:
struct itimerval {
struct timeval it_interval;
struct timeval it_value;
};
struct timeval {
long tv_sec;
long tv_usec;
};
第一个参数which,定时器类型:
ITIMER_REAL : 以系统真实的时间来计算,它送出SIGALRM信号。
ITIMER_VIRTUAL : 以该进程在用户态下花费的时间来计算,它送出SIGVTALRM信号。
ITIMER_PROF : 以该进程在用户态下和内核态下所费的时间来计算,它送出SIGPROF信号。
第二个参数value,指定间隔时间
第三个参数ovalue,用来返回上一次定时器的间隔时间,可设为NULL。
it_interval:指定间隔时间
it_value:指定初始定时时间。
1、如果只指定it_value,it_interval为NULL,就是实现一次定时;
2、如果同时指定 it_interval,则超时后,系统会重新初始化it_value为it_interval,实现重复定时;
3、两者都清零,则会清除定时器。
tv_sec提供秒级精度,tv_usec提供微秒级精度,以值大的为先,注意1s = 1000000us。
#include <pthread.h>
#include <stdlib.h>
#include <stdint.h>
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
#include <signal.h>
#include <sys/time.h> //struct itimerval, setitimer()
void timer_cb(int sig)
{
static int count = 0;
if(SIGALRM == sig)
{
printf("timer %d\n", ++count);
}
}
int main(void)
{
struct itimerval tick;
signal(SIGALRM, timer_cb);
memset(&tick, 0, sizeof(tick));
//设置第一次超时时间
tick.it_value.tv_sec = 0;
tick.it_value.tv_usec = 500;
//设置首次之后,间隔时间
tick.it_interval.tv_sec = 1;
tick.it_interval.tv_usec = 1000;
if(setitimer(ITIMER_REAL, &tick, NULL) < 0)
printf("Set timer failed!\n");
getchar();
return 0;
}
-----------------------------------------仅此记录----------------------------------------------