最强悍的定时器插口来自POSIX时钟系,其创建、初始化以及删掉一个定时器的行动被分为三个不同的函数:timer_create()(创建定时器)、timer_settime()(初始化定时器)以及timer_delete(销毁它)。

创建一个定时器:

<span style="font-family:SimSun;font-size:18px;color:#ff0000">int timer_create(clockid_t clock_id, struct sigevent *evp, timer_t *timerid)</span>

登录后复制

进程可以通过调用timer_create()创建特定的定时器,定时器是每位进程自己的,不是在fork时承继的。clock_id说明定时器是基于那个时钟的,*timerid装载的是被创建的定时器的ID。该函数创建了定时器,并将他的ID装入timerid指向的位置中。参数evp指定了定时器到期要形成的异步通知。假如evp为NULL,这么定时器到期会形成默认的讯号,对CLOCK_REALTIMER来说,默认讯号就是SIGALRM。假如要形成除默认讯号之外的其它讯号linux,程序必须将evp->sigev_signo设置为期望的信号码。structsigevent结构中的成员evp->sigev_notify说明了定时器到期时应当采取的行动。一般,这个成员的值为SIGEV_SIGNAL,这个值说明在定时器到期时,会形成一个讯号。程序可以将成员evp->sigev_notify设为SIGEV_NONE来避免定时器到期时形成讯号。

假如几个定时器形成了同一个讯号,处理程序可以用evp->sigev_value来分辨是那个定时器形成了讯号。要实现这些功能,程序必须在为讯号安装处理程序时,使用structsigaction的成员sa_flags中的标志符SA_SIGINFO。

clock_id取值为以下:

struct sigevent
{
<span>	</span>int sigev_notify; //notification type
<span>	</span>int sigev_signo; //signal number
<span>	</span>union sigval sigev_value; //signal value
<span>	</span>void (*sigev_notify_function)(union sigval);
<span>	</span>pthread_attr_t *sigev_notify_attributes;
}
union sigval
{
<span>	</span>int sival_int; //integer value
<span>	</span>void *sival_ptr; //pointer value
}

登录后复制

通过将evp->sigev_notify设定为如下值来定制定时器到期后的行为:

启动一个定时器:

timer_create()所创建的定时器并未启动。要将它关联到一个到期时间以及启动时钟周期,可以使用timer_settime()。

<span style="font-family:SimSun;font-size:18px;color:#ff0000">int timer_settime(timer_t timerid, int flags, const struct itimerspec *value, struct itimerspect *ovalue);</span>

登录后复制

struct itimespec
{
struct timespec it_interval; 
struct timespec it_value; 
}; 

登录后复制

就像settimer(),it_value用于指定当前的定时器到期时间。当定时器到期,it_value的值会被更新成it_interval的值。假如it_interval的值为0,则定时器不是一个时间间隔定时器,一旦it_value到期都会回到未启动状态。timespec的结构提供了毫秒级码率:

struct timespec
{
time_t tv_sec;
long tv_nsec;
};

登录后复制

假如flags的值为TIMER_ABSTIME,则value所指定的时间值会被剖析成绝对值(此值的默认的剖析方法为相对于当前的时间)。这个经更改的行为可防止取得当前时间、计算“该时间”与“所期望的未来时间”的相对差额以及启动定时器期间导致竞争条件。

定时器程序设计实验报告_linux 定时器程序_定时器程序编写

假如ovalue的值不是NULL,则之前的定时器到期时间会被存入其所提供的itimerspec。假如定时器之前处在未启动状态arm linux,则此结构的成员全就会被设定成0。

获得一个活动定时器的剩余时间:

<span style="font-family:SimSun;font-size:18px;color:#ff0000">int timer_gettime(timer_t timerid,struct itimerspec *value);</span>

登录后复制

取得一个定时器的超限运行次数:

有可能一个定时器到期了,而同一定时器上一次到期时形成的讯号还处于挂起状态。在这些情况下,其中的一个讯号可能会遗失。这就是定时器超限。程序可以通过调用timer_getoverrun来确定一个特定的定时器出现这些超限的次数。定时器超限只能发生在同一个定时器形成的讯号上。由多个定时器linux 定时器程序,甚至是这些使用相同的时钟和讯号的定时器,所形成的讯号就会排队而不会遗失。

<span style="font-family:SimSun;font-size:18px;color:#ff0000">int timer_delete (timer_t timerid);</span>

登录后复制登录后复制

执行成功时,timer_getoverrun()会返回定时器初次到期与通知进程(比如通过讯号)定时器已到期之间额外发生的定时器到期次数。举例来说linux 定时器程序,在我们之前的事例中,一个1ms的定时器运行了10ms,则此调用会返回9。假如超限运行的次数等于或小于DELAYTIMER_MAX,则此调用会返回DELAYTIMER_MAX。

执行失败时,此函数会返回-1并将errno设定会EINVAL,这个惟一的错误情况代表timerid指定了无效的定时器。

删掉一个定时器:

<span style="font-family:SimSun;font-size:18px;color:#ff0000">int timer_delete (timer_t timerid);</span>

登录后复制登录后复制

一次成功的timer_delete()调用会销毁关联到timerid的定时器而且返回0。执行失败时,此调用会返回-1并将errno设定会EINVAL,这个惟一的错误情况代表timerid不是一个有效的定时器。

例1:

voidhandle()

time_tt;

charp[32];

time(&t);

strftime(p,sizeof(p),”%T”,localtime(&t));

printf(“timeis%s/n”,p);

intmain()

structsigeventevp;

structitimerspects;

timer_ttimer;

intret;

evp.sigev_value.sival_ptr=&timer;

evp.sigev_notify=SIGEV_SIGNAL;

evp.sigev_signo=SIGUSR1;

定时器程序设计实验报告_定时器程序编写_linux 定时器程序

signal(SIGUSR1,handle);

ret=timer_create(CLOCK_REALTIME,&evp,&timer);

if(ret)

perror(“timer_create”);

_sec=1;

_nsec=0;

_sec=3;

_nsec=0;

ret=timer_settime(timer,0,&ts,NULL);

if(ret)

perror(“timer_settime”);

while(1);

例2:

voidhandle(unionsigvalv)

time_tt;

charp[32];

time(&t);

strftime(p,sizeof(p),”%T”,localtime(&t));

printf(“%sthread%lu,val=%d,signalcaptured./n”,p,pthread_self(),v.sival_int);

return;

intmain()

structsigeventevp;

structitimerspects;

timer_ttimer;

intret;

memset(&evp,0,sizeof(evp));

定时器程序设计实验报告_linux 定时器程序_定时器程序编写

evp.sigev_value.sival_ptr=&timer;

evp.sigev_notify=SIGEV_THREAD;

evp.sigev_notify_function=handle;

evp.sigev_value.sival_int=3;//作为handle()的参数

ret=timer_create(CLOCK_REALTIME,&evp,&timer);

if(ret)

perror(“timer_create”);

_sec=1;

_nsec=0;

_sec=3;

_nsec=0;

ret=timer_settime(timer,TIMER_ABSTIME,&ts,NULL);

if(ret)

perror(“timer_settime”);

while(1);

以上就是最强大的定时器接口来自POSIX时钟系列,你知道吗?的详细内容,更多请关注小闻网其它相关文章!

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。