Linux中用户态可以使用的计时器主要有alarm和timer.timer可以提供更加精确的时间,alarm只能精确到秒。
#include <sys/time.h> int getitimer(int which, struct itimerval *curr_value); int setitimer(int which, const struct itimerval *new_value,struct itimerval *old_value); struct itimerval { struct timeval it_interval; /* next value */ struct timeval it_value; /* current value */ }; struct timeval { long tv_sec; /* seconds */ long tv_usec; /* microseconds */ };
通过函数setitimer可以设定计时器,which指定了计算时间的细节,包括:
ITIMER_REAL 真正的时间,发出SIGALRM信号。
ITIMER_VIRTUAL 只有当进程运行时才会计时,SIGVTALRM。
ITIMER_PROF:减少进程的有效时间和系统时间(为进程调度用的时间).这个经常和上面一个使用用来计算系统内核时间和用户时间.产生SIGPROF信号.
在结构体itimerval中的it_value代表了当前计时器剩余的时间,如果为0,表示计时器停止。it_interval表示计时器到零时,下一次的初始值。通过setitimer函数将it_value设置为0表示计时器停止,it_interval设置为0表示当前计时器到0后就不再计时。
fork后子进程不能继承,exec可以继承该计时器。
示例代码:
#include <sys/time.h> #include <signal.h> #include <stdio.h> void sig_timer(int signo) { if(signo == SIGVTALRM) { printf("time over/n"); exit(0); } else printf("other signo/n"); } int main() { struct itimerval timer = {0}; timer.it_value.tv_sec = 5; timer.it_value.tv_usec = 10; timer.it_interval.tv_sec = 0; timer.it_interval.tv_usec = 0; struct sigaction st_sig = {0}; st_sig.sa_handler = sig_timer; if(sigaction(SIGVTALRM ,&st_sig,NULL) < 0) printf("error to set sig proc/n"); if(setitimer(ITIMER_VIRTUAL,&timer,NULL) < 0) { printf("error to set timer/n"); } sleep(10); printf("sleep back/n"); return 0; }
//实例中使用ITIMER_VIRTUAL这个时候计时器的时间由于进程在sleep,所以没有计算,最后通过sleep back退出。如果设置为ITIMER_REAL则会显示time over。从信号处理程序中返回。