免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 5747 | 回复: 8
打印 上一主题 下一主题

利用kqueue在单进程中实现多定时器 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-11-16 22:31 |只看该作者 |倒序浏览
利用C语言函数,一般在一个进程中只能有一个定时器。要想实现多定时器,一般是在这个定时器中使用累计计数来实现。当定时器比较多,和定时间隔差别很大,如0.02秒和10秒,计数就带来了很大的额外开销。
  当然也可用循环嵌入sleep,nanosleep的方法,但是这增加了循环内不必要的阻塞。
  多进程和多线程的方法,增加了程序的复杂度和开销。

  利用FreeBSD的kqueue可以在单进程中实现多定时器, 同时避免以上弊端。
  下面是我写的一小段代码,与大家共享,献丑了,呵呵。转载请标明来自CU。

  
   


  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <errno.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/event.h>
  7. #include <sys/time.h>

  8. int main()
  9. {
  10.   struct kevent changes[3];
  11.   struct kevent events[3];

  12.   int kq=kqueue();
  13.   if(kq==-1)
  14.   {
  15.     fprintf(stderr,"Kqueue error: %s\n",strerror(errno));
  16.     return -1;
  17.   }

  18.   struct timespec thetime;
  19.   bzero(&thetime,sizeof(thetime));

  20.   
  21.   EV_SET(&changes[0],0,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 5000, 0);
  22.   EV_SET(&changes[1],1,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 2000, 0);
  23.   EV_SET(&changes[2],2,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 1000, 0);

  24.   if(kevent(kq,changes,3,NULL,0,&thetime)==-1)
  25.   {
  26.      fprintf(stderr,"kevent error: %s\n",strerror(errno));
  27.      return -1;
  28.   }

  29.   for(;;)
  30.   {
  31.     int nev=kevent(kq,NULL,0,events,1,&thetime);
  32.    
  33.     if(nev==-1)
  34.     {
  35.       fprintf(stderr,"kevent error: %s\n",strerror(errno));
  36.       return -1;
  37.     }
  38.   
  39.     if(nev>0)
  40.     {
  41.       int i;
  42.       for(i=0;i<nev;i++)
  43.       {

  44.         switch(events[i].ident)
  45.         {
  46.            case 0:
  47.              fprintf(stdout, "This is 5 sec timer.\n");
  48.              break;
  49.            case 1:
  50.              fprintf(stdout, "This is 2 sec timer.\n");
  51.              break;
  52.            case 2:
  53.              fprintf(stdout, "This is 1 sec timer.\n");
  54.              break;
  55.         }
  56.       }
  57.     }

  58.    /*
  59.          ----------可以做其他事情。
  60.    */

  61.   }
  62.   
  63.   return 0;
  64. }


复制代码

[ 本帖最后由 doctorjxd 于 2007-11-16 22:32 编辑 ]

评分

参与人数 1可用积分 +4 收起 理由
gvim + 4 感谢分享心得。

查看全部评分

论坛徽章:
0
2 [报告]
发表于 2007-11-19 09:14 |只看该作者
不错! 有无kqueue 更深入的文档?

论坛徽章:
0
3 [报告]
发表于 2007-11-19 11:26 |只看该作者
原帖由 benjiam 于 2007-11-19 09:14 发表
不错! 有无kqueue 更深入的文档?


http://wiki.netbsd.se/kqueue_tutorial

要在freebsd上编译kqueue的例子 要加头文件 <sys/types.h>

论坛徽章:
0
4 [报告]
发表于 2007-12-29 09:15 |只看该作者

讨论一下

这样的定时能准确吗?我在FreeBSD6.3上测试好像不太准,

我觉得这个定时受程序中做其他事情的影响,如果做的其他事情时间很长,这个定时
不就不起作用了吗?
我在
   /*
         ----------可以做其他事情。
   */

加了个sleep(10)试了。

论坛徽章:
0
5 [报告]
发表于 2007-12-29 15:18 |只看该作者
用epoll能实现这个吗?怎样实现?

论坛徽章:
0
6 [报告]
发表于 2007-12-29 15:19 |只看该作者
原帖由 jameszxj 于 2007-12-29 09:15 发表
这样的定时能准确吗?我在FreeBSD6.3上测试好像不太准,

我觉得这个定时受程序中做其他事情的影响,如果做的其他事情时间很长,这个定时
不就不起作用了吗?
我在
   /*
         ----------可以做其他 ...


以下是我的相法,说得不对请指教:
1, 我认为这是你的用法不对。我认为楼主给的例子相当于一个信号处理函数。在这种类型的函数里就不应该做复杂的操作,越简单越好,最好是仅仅设个标志位之类的。

2,即使你做其它事情的时间很长也不会影响定时的准确性(下面的程序测试为证),只不过有可能你自己来不及及时响应。但正如我在第一点里指出的,那是你自己的问题。


  1. /home/lgfang/tmp $ ./a.out
  2. 1198911901 start
  3. 1198911904 3 sec timer.
  4. 1198911907 3 sec timer.
  5. 1198911909 8 sec timer.
  6. 1198911910 3 sec timer.
  7. 1198911913 3 sec timer.
  8. 1198911916 3 sec timer.
  9. 1198911917 8 sec timer.
  10. 1198911919 3 sec timer.
  11. 1198911922 3 sec timer.
  12. 1198911925 8 sec timer.
  13. 1198911926 3 sec timer.
  14. 1198911928 3 sec timer.
  15. 1198911931 3 sec timer.
  16. 1198911933 8 sec timer.
复制代码


  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <errno.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/event.h>
  7. #include <sys/time.h>

  8. int main()
  9. {
  10.   struct kevent changes[3];
  11.   struct kevent events[3];

  12.   int kq=kqueue();
  13.   if(kq==-1)
  14.     {
  15.       fprintf(stderr,"Kqueue error: %s\n",strerror(errno));
  16.       return -1;
  17.     }

  18.   struct timespec thetime;
  19.   bzero(&thetime,sizeof(thetime));

  20.   EV_SET(&changes[0],0,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 8000, 0);
  21.   EV_SET(&changes[1],1,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 3000, 0);

  22.   if(kevent(kq,changes,2,NULL,0,&thetime)==-1)
  23.     {
  24.       fprintf(stderr,"kevent error: %s\n",strerror(errno));
  25.       return -1;
  26.     }
  27.   printf ("%d start\n", time(NULL));

  28.   for(;;)
  29.     {
  30.       time_t t = time(NULL);
  31.       int nev=kevent(kq,NULL,0,events,1,NULL);

  32.       if(nev==-1)
  33.         {
  34.           fprintf(stderr,"kevent error: %s\n",strerror(errno));
  35.           return -1;
  36.         }

  37.       if(nev>0)
  38.         {
  39.           int i;
  40.           for(i=0;i<nev;i++)
  41.             {

  42.               switch(events[i].ident)
  43.                 {
  44.                 case 0:
  45.                   fprintf(stdout, "%d 8 sec timer.\n", t);
  46.                   break;
  47.                 case 1:
  48.                   fprintf(stdout, "%d 3 sec timer.\n", t);
  49.                   break;
  50.                 }
  51.             }
  52.         }

  53.       sleep (1);

  54.     }

  55.   return 0;
  56. }
复制代码

[ 本帖最后由 lgfang 于 2007-12-29 19:01 编辑 ]

论坛徽章:
0
7 [报告]
发表于 2007-12-29 20:02 |只看该作者

论坛徽章:
0
8 [报告]
发表于 2007-12-29 21:18 |只看该作者
epoll好像不能实现
EV_SET(&changes[1],1,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 2000, 0);
epoll没有EVFILT_TIMER这个参数,不一样~~

论坛徽章:
0
9 [报告]
发表于 2007-12-31 19:26 |只看该作者
原帖由 lgfang 于 2007-12-29 15:19 发表


以下是我的相法,说得不对请指教:
1, 我认为这是你的用法不对。我认为楼主给的例子相当于一个信号处理函数。在这种类型的函数里就不应该做复杂的操作,越简单越好,最好是仅仅设个标志位之类的。

2, ...



楼主的注释和标题都写明了 “ 利用FreeBSD的kqueue可以在单进程中实现多定时器, 同时避免以上弊端。”
如果只是说信号处理,我就不研究定时的问题了。

你说的准因为你sleep是1,如果是10呢?

我主要在嵌入式上编程,嵌入式系统的定时器都是互不影响的,现在转到*unix下,对于信号会影响sleep等系统调用很不习惯。
而且一个进程只有一个ITIMER_REAL定时器,我是想能在*nix下找到一个比较好的解决方法。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP