一,三种时间结构

time_t://seconds

 

struct timeval {

long tv_sec; /* seconds */

long tv_usec; /* microseconds */

};

 

struct timespec {

time_t tv_sec; /* seconds */

long tv_nsec; /* nanoseconds */

};

二,setitimer()

现在的系统中很多程序不再使用alarm调用,而是使用setitimer调用来设置定时器,用getitimer来得到定时器的状态,

这两个调用的声明格式如下:

#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);

参数:

  • 第一个参数which指定定时器类型
  • 第二个参数是结构itimerval的一个实例,结构itimerval形式
  • 第三个参数可不做处理。

返回值:成功返回0失败返回-1

该系统调用给进程提供了三个定时器,它们各自有其独有的计时域,当其中任何一个到达,就发送一个相应的信号给进程,并使得计时器重新开始。三个计时器由参数which指定,如下所示:

TIMER_REAL:按实际时间计时,计时到达将给进程发送SIGALRM信号。

ITIMER_VIRTUAL:仅当进程执行时才进行计时。计时到达将发送SIGVTALRM信号给进程。

ITIMER_PROF:当进程执行时和系统为该进程执行动作时都计时。与ITIMER_VIR-TUAL是一对,该定时器经常用来统计进程在用户态和内核态花费的时间。计时到达将发送SIGPROF信号给进程。

定时器中的参数value用来指明定时器的时间,其结构如下:

struct itimerval {

        struct timeval it_interval; /* 第一次之后每隔多长时间 */

        struct timeval it_value; /* 第一次调用要多长时间 */

};

该结构中timeval结构定义如下:

struct timeval {

        long tv_sec; /* 秒 */

        long tv_usec; /* 微秒,1秒 = 1000000 微秒*/

};

在setitimer 调用中,参数ovalue如果不为空,则其中保留的是上次调用设定的值。定时器将it_value递减到0时,产生一个信号,并将it_value的值设定为it_interval的值,然后重新开始计时,如此往复。当it_value设定为0时,计时器停止,或者当它计时到期,而it_interval 为0时停止。调用成功时,返回0;错误时,返回-1,并设置相应的错误代码errno:

EFAULT:参数value或ovalue是无效的指针。

EINVAL:参数which不是ITIMER_REAL、ITIMER_VIRT或ITIMER_PROF中的一个。

示例一:

  1. #include <unistd.h>
  2. #include <sys/stat.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <fcntl.h>
  6.  
  7. #include <stdlib.h>
  8. #include <stdio.h>
  9. #include <errno.h>
  10. #include <string.h>
  11. #include <signal.h>
  12. #include <sys/time.h>
  13.  
  14. #define ERR_EXIT(m) \
  15. do \
  16. { \
  17. perror(m); \
  18. exit(EXIT_FAILURE); \
  19. } while(0)
  20.  
  21. void handler(int sig)
  22. {
  23. printf("recv a sig=%d\n", sig);
  24. }
  25.  
  26. int main(int argc, char *argv[])
  27. {
  28. if (signal(SIGALRM, handler) == SIG_ERR)
  29. ERR_EXIT("signal error");
  30.  
  31. struct timeval tv_interval = {1, 0};
  32. struct timeval tv_value = {5, 0};
  33. struct itimerval it;
  34. it.it_interval = tv_interval;
  35. it.it_value = tv_value;
  36. setitimer(ITIMER_REAL, &it, NULL);
  37.  
  38. for (;;)
  39. pause();
  40. return 0;
  41. }

结果:

可以看到第一次发送信号是在5s以后,之后每隔一秒发送一次信号

示例二:获得产生时钟信号的剩余时间

  1. #include <unistd.h>
  2. #include <sys/stat.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <fcntl.h>
  6.  
  7. #include <stdlib.h>
  8. #include <stdio.h>
  9. #include <errno.h>
  10. #include <string.h>
  11. #include <signal.h>
  12. #include <sys/time.h>
  13.  
  14. #define ERR_EXIT(m) \
  15. do \
  16. { \
  17. perror(m); \
  18. exit(EXIT_FAILURE); \
  19. } while(0)
  20.  
  21. int main(int argc, char *argv[])
  22. {
  23. struct timeval tv_interval = {1, 0};
  24. struct timeval tv_value = {1, 0};
  25. struct itimerval it;
  26. it.it_interval = tv_interval;
  27. it.it_value = tv_value;
  28. setitimer(ITIMER_REAL, &it, NULL);
  29.  
  30. int i;
  31. for (i=0; i<10000; i++);
  32.  
  33. //第一种方式获得剩余时间
  34. struct itimerval oit;
  35. setitimer(ITIMER_REAL, &it, &oit);//利用oit获得剩余时间产生时钟信号
  36. printf("%d %d %d %d\n", (int)oit.it_interval.tv_sec, (int)oit.it_interval.tv_usec, (int)oit.it_value.tv_sec, (int)oit.it_value.tv_usec);
  37. //第二种方式获得剩余时间
  38. //getitimer(ITIMER_REAL, &it);
  39. //printf("%d %d %d %d\n", (int)it.it_interval.tv_sec, (int)it.it_interval.tv_usec, (int)it.it_value.tv_sec, (int)it.it_value.tv_usec);
  40.  
  41. return 0;
  42. }

结果:

用第一种方式:

用第二种方式:利用getitimer在不重新设置时钟的情况下获取剩余时间

剩余时间是指:距离下一次调用定时器产生信号所需时间,这里由于for循环不到一秒就执行完,定时器还来不及产生时钟信号,所以有剩余时间

示例三:每隔一秒发出一个SIGALRM,每隔0.5秒发出一个SIGVTALRM信号

  1. #include <signal.h>
  2. #include <unistd.h>
  3. #include <stdio.h>
  4. #include <sys/time.h>
  5.  
  6. void sigroutine(int signo)
  7. {
  8. switch (signo) {
  9. case SIGALRM:
  10. printf("Catch a signal -- SIGALRM\n ");
  11. break;
  12. case SIGVTALRM:
  13. printf("Catch a signal -- SIGVTALRM\n ");
  14. break;
  15. }
  16. return;
  17. }
  18.  
  19. int main()
  20. {
  21.  
  22. struct itimerval value,value2;
  23. printf("process id is %d\n ",getpid());
  24.  
  25. signal(SIGALRM, sigroutine);
  26.  
  27. signal(SIGVTALRM, sigroutine);
  28.  
  29. value.it_value.tv_sec = 1;
  30.  
  31. value.it_value.tv_usec = 0;
  32.  
  33. value.it_interval.tv_sec = 1;
  34.  
  35. value.it_interval.tv_usec = 0;
  36.  
  37. setitimer(ITIMER_REAL, &value,NULL);
  38.  
  39. value2.it_value.tv_sec = 0;
  40.  
  41. value2.it_value.tv_usec = 500000;
  42.  
  43. value2.it_interval.tv_sec = 0;
  44.  
  45. value2.it_interval.tv_usec = 500000;
  46.  
  47. setitimer(ITIMER_VIRTUAL, &value2,NULL);
  48.  
  49. for (;;) ;
  50.  
  51. }

结果:

可知确实是没两次SIGVTALRM一次SIGALRM

linux系统编程之信号(八):三种时间结构及定时器setitimer()详解的更多相关文章

  1. linux系统编程之信号(三):信号安装、signal、kill,arise讲解

    一,信号安装 如果进程要处理某一信号,那么就要在进程中安装该信号.安装信号主要用来确定信号值及进程针对该信号值的动作之间的映射关系,即进程将要处理哪个信号:该信号被传递给进程时,将执行何种操作. li ...

  2. linux系统编程之信号(七)

    今天继续学习信号,主要是学习关于时间和定时器相关的函数的使用,关于这个实际上有很多内容,这里先简要进行说明,等之后再慢慢进行相关深入,也主要是为接下来要做的一个综合linux系统编程的例子做准备,好了 ...

  3. linux系统编程之信号(一)

    今天起,开始新的知识的学习,对于上个系列进程的学习还差一个理论上的总结,这个会下次补回来,以便通过实践之后,再用理论将其巩固一下,好了,话不多说,开始进入这个主题的学习----信号,很重要,但不是太容 ...

  4. linux系统编程之信号(一):中断与信号

    一,什么是中断? 1.中断的基本概念 中断是指计算机在执行期间,系统内发生任何非寻常的或非预期的急需处理事件,使得CPU暂时中断当前正在执行的程序而转去执行相应的事件处理程序,待处理完毕后又返回原来被 ...

  5. linux系统中,文件的三种特殊权限

    背景介绍 在linux系统中,我们熟知有rwx三种权限,对应所有者,同组用户,其他用户三种用户的权限,一共9个位来指定一个文件的权限情况,通过chmod xxx 来更改权限属性,其中xxx是已八进制表 ...

  6. linux系统root密码忘了怎么办 三种方法快速找回root密码

    linux root密码找回方法一 第1步:在系统进入单用户状态,直接用passwd root去更改. 第2步:用安装光盘引导系统,进行linux rescue状态,将原来/分区挂接上来,作法如下: ...

  7. linux系统编程之信号(二):信号处理流程(产生、注册、注销、执行)

        对于一个完整的信号生命周期(从信号发送到相应的处理函数执行完毕)来说,可以分为三个阶段: 信号诞生 信号在进程中注册 信号在进程中的注销 信号处理函数执行 1    信号诞生     信号事件 ...

  8. linux系统编程之进程(三):进程复制fork,孤儿进程,僵尸进程

    本节目标: 复制进程映像 fork系统调用 孤儿进程.僵尸进程 写时复制 一,进程复制(或产生)      使用fork函数得到的子进程从父进程的继承了整个进程的地址空间,包括:进程上下文.进程堆栈. ...

  9. linux系统编程之信号(二)

    经过了漫长的间歇,对于c语言的学习也被中断了很久,现实确实有很多的无耐,计划中的事情总会被打乱,但不管怎样,学习的道路是不能休止的,所以经过了一断温习后现在继续学习C语言,话不多说,进入正题: 信号分 ...

随机推荐

  1. (转) Ringbuffer为什么这么快?

    原文地址:http://ifeve.com/ringbuffer/ 最近,我们开源了LMAX Disruptor,它是我们的交易系统吞吐量快(LMAX是一个新型的交易平台,号称能够单线程每秒处理数百万 ...

  2. java内存模型:简单理解

    1.Java内存模型(Java Memory Model,JMM) 2.JMM定义了线程和主内存之间的抽象关系: 线程之间的共享变量存储在主内存(main memory)中,每个线程都有一个私有的本地 ...

  3. Web服务技术协议:REST与SOAP

    Web服务技术就有SOAP(Simple Object Access Protocol,简单对象访问协议)和REST(Representational State Transfer,表示性状态转移) ...

  4. 【校招面试 之 剑指offer】第9-2题 用两个队列实现一个栈

    #include<iostream> #include<queue> using namespace std; // 对于出栈解决的思路是:将queue1的元素除了最后一个外全 ...

  5. 安装Oracle客户端寻找配置文件tnsnames.ora

    # tnsnames.ora Network Configuration File: D:\app\Administrator\product\11.2.0\dbhome_1\network\admi ...

  6. 利用redis完成自动补全搜索功能(二)

    前面介绍了自动完成的大致思路,现在把搜索次数的功能也结合上去.我采用的是hash表来做的,当然也可以在生成分词的时候,另外一个有序集合来维护排序, 然后2个有序集合取交集即可.这里介绍hash的方式来 ...

  7. vmware14中安装centos7并使用docker发布spring-boot项目

    1.vmare中centos7安装(同一路由器无线网络下) 1.1选择桥接模式 1.2修改配置文件 vi /etc/sysconfig/network-scripts/ifcfg-ens33(这里不一 ...

  8. 在 Microsoft Dynamics 365 Online中如何调试Plugins in

    How to debug plugins in Microsoft Dynamics 365 Online 调试方式请查阅https://www.linkedin.com/pulse/how-debu ...

  9. 品味性能之道<七>:索引基础

    一.索引概述      索引(index),它是数据库必不可少的一部分.它其实很简单呐!很好理解.      索引好比如一本书的目录,一张地图,一个写字楼里挂在大堂墙上的公司名录,一个地铁站的出口指示 ...

  10. Java中通过SimpleDateFormat格式化当前时间:/** 输出格式:20060101010101001**/

    import java.util.*; import java.text.SimpleDateFormat; int y,m,d,h,mi,s,ms; String cur; Calendar cal ...