在linux下如果对定时要求不太精确的话,使用alarm()和signal()就行了;
但是如果想要实现精度较高的定时功能的话,就要使用setitimer函数。
核心api:
int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue);
调用成功返回0,否则返回-1;
which为定时器类型,setitimer支持3种类型的定时器:
ITIMER_REAL: 以系统真实的时间来计算,它送出SIGALRM信号。
ITIMER_VIRTUAL: -以该进程在用户态下花费的时间来计算,它送出SIGVTALRM信号。
ITIMER_PROF: 以该进程在用户态下和内核态下所费的时间来计算,它送出SIGPROF信号。
第一个参数which指定定时器类型(上面三种之一);第二个参数是结构itimerval的一个实例;第三个参数可不做处理。
//下面这两个是八股,要用计时器setitimer()就必须先配好struct itimerval {
struct timeval it_interval; //it_interval指定间隔时间
struct timeval it_value; //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 = 1000ms//ovalue用来保存先前的值,常设为NULL。
例子:
#include #include #include #include #include #include void call_back(int sig){static int count = 1;printf("call_back : %d
", count++);}int main(void){signal(SIGALRM, call_back);struct itimerval new_value;memset(&new_value, 0, sizeof(new_value));new_value.it_interval.tv_sec = 1; //设置再次定时时间为1Snew_value.it_interval.tv_usec = 0;new_value.it_value.tv_sec = 2; //设置首次定时时间为2Snew_value.it_value.tv_usec = 0;int ret = setitimer(ITIMER_REAL, &new_value, NULL);if (0 != ret){perror("setitimer");exit(-1);}getchar();return 0;}
线程安全:
定时器死了,死循环就停不了,线程退出不了,还有就是主程序退出时,你的这个线程怎么安全退出。
还有就是实际的延时的问题,线程实际的输出时间比设定的长或者短。
这里要特别注意一下sleep()和大写的Sleep()还有usleep()的时间单位是不同的,秒,毫秒和微秒