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。

SIGVTALRM 包含了进程在内核和用户态运行的时间。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。从信号处理程序中返回。

Logo

更多推荐