在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 <stdio.h>
- #include <sys/time.h>
- #include <unistd.h>
- #include <stdlib.h>
- #include <string.h>
- #include <signal.h>
-
- 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; //设置再次定时时间为1S
- new_value.it_interval.tv_usec = 0;
- new_value.it_value.tv_sec = 2; //设置首次定时时间为2S
- new_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()的时间单位是不同的,秒,毫秒和微秒