亚洲av成人无遮挡网站在线观看,少妇性bbb搡bbb爽爽爽,亚洲av日韩精品久久久久久,兔费看少妇性l交大片免费,无码少妇一区二区三区

  免費(fèi)注冊(cè) 查看新帖 |

Chinaunix

  平臺(tái) 論壇 博客 文庫
最近訪問板塊 發(fā)新帖
查看: 5766 | 回復(fù): 8
打印 上一主題 下一主題

利用kqueue在單進(jìn)程中實(shí)現(xiàn)多定時(shí)器 [復(fù)制鏈接]

論壇徽章:
0
跳轉(zhuǎn)到指定樓層
1 [收藏(0)] [報(bào)告]
發(fā)表于 2007-11-16 22:31 |只看該作者 |倒序?yàn)g覽
利用C語言函數(shù),一般在一個(gè)進(jìn)程中只能有一個(gè)定時(shí)器。要想實(shí)現(xiàn)多定時(shí)器,一般是在這個(gè)定時(shí)器中使用累計(jì)計(jì)數(shù)來實(shí)現(xiàn)。當(dāng)定時(shí)器比較多,和定時(shí)間隔差別很大,如0.02秒和10秒,計(jì)數(shù)就帶來了很大的額外開銷。
  當(dāng)然也可用循環(huán)嵌入sleep,nanosleep的方法,但是這增加了循環(huán)內(nèi)不必要的阻塞。
  多進(jìn)程和多線程的方法,增加了程序的復(fù)雜度和開銷。

  利用FreeBSD的kqueue可以在單進(jìn)程中實(shí)現(xiàn)多定時(shí)器, 同時(shí)避免以上弊端。
  下面是我寫的一小段代碼,與大家共享,獻(xiàn)丑了,呵呵。轉(zhuǎn)載請(qǐng)標(biāo)明來自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. }


復(fù)制代碼

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

評(píng)分

參與人數(shù) 1可用積分 +4 收起 理由
gvim + 4 感謝分享心得。

查看全部評(píng)分

論壇徽章:
0
2 [報(bào)告]
發(fā)表于 2007-11-19 09:14 |只看該作者
不錯(cuò)! 有無kqueue 更深入的文檔?

論壇徽章:
0
3 [報(bào)告]
發(fā)表于 2007-11-19 11:26 |只看該作者
原帖由 benjiam 于 2007-11-19 09:14 發(fā)表
不錯(cuò)! 有無kqueue 更深入的文檔?


http://wiki.netbsd.se/kqueue_tutorial

要在freebsd上編譯kqueue的例子 要加頭文件 <sys/types.h>

論壇徽章:
0
4 [報(bào)告]
發(fā)表于 2007-12-29 09:15 |只看該作者

討論一下

這樣的定時(shí)能準(zhǔn)確嗎?我在FreeBSD6.3上測(cè)試好像不太準(zhǔn),

我覺得這個(gè)定時(shí)受程序中做其他事情的影響,如果做的其他事情時(shí)間很長,這個(gè)定時(shí)
不就不起作用了嗎?
我在
   /*
         ----------可以做其他事情。
   */

加了個(gè)sleep(10)試了。

論壇徽章:
0
5 [報(bào)告]
發(fā)表于 2007-12-29 15:18 |只看該作者
用epoll能實(shí)現(xiàn)這個(gè)嗎?怎樣實(shí)現(xiàn)?

論壇徽章:
0
6 [報(bào)告]
發(fā)表于 2007-12-29 15:19 |只看該作者
原帖由 jameszxj 于 2007-12-29 09:15 發(fā)表
這樣的定時(shí)能準(zhǔn)確嗎?我在FreeBSD6.3上測(cè)試好像不太準(zhǔn),

我覺得這個(gè)定時(shí)受程序中做其他事情的影響,如果做的其他事情時(shí)間很長,這個(gè)定時(shí)
不就不起作用了嗎?
我在
   /*
         ----------可以做其他 ...


以下是我的相法,說得不對(duì)請(qǐng)指教:
1, 我認(rèn)為這是你的用法不對(duì)。我認(rèn)為樓主給的例子相當(dāng)于一個(gè)信號(hào)處理函數(shù)。在這種類型的函數(shù)里就不應(yīng)該做復(fù)雜的操作,越簡單越好,最好是僅僅設(shè)個(gè)標(biāo)志位之類的。

2,即使你做其它事情的時(shí)間很長也不會(huì)影響定時(shí)的準(zhǔn)確性(下面的程序測(cè)試為證),只不過有可能你自己來不及及時(shí)響應(yīng)。但正如我在第一點(diǎn)里指出的,那是你自己的問題。


  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.
復(fù)制代碼


  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. }
復(fù)制代碼

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

論壇徽章:
0
7 [報(bào)告]
發(fā)表于 2007-12-29 20:02 |只看該作者

論壇徽章:
0
8 [報(bào)告]
發(fā)表于 2007-12-29 21:18 |只看該作者
epoll好像不能實(shí)現(xiàn)
EV_SET(&changes[1],1,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 2000, 0);
epoll沒有EVFILT_TIMER這個(gè)參數(shù),不一樣~~

論壇徽章:
0
9 [報(bào)告]
發(fā)表于 2007-12-31 19:26 |只看該作者
原帖由 lgfang 于 2007-12-29 15:19 發(fā)表


以下是我的相法,說得不對(duì)請(qǐng)指教:
1, 我認(rèn)為這是你的用法不對(duì)。我認(rèn)為樓主給的例子相當(dāng)于一個(gè)信號(hào)處理函數(shù)。在這種類型的函數(shù)里就不應(yīng)該做復(fù)雜的操作,越簡單越好,最好是僅僅設(shè)個(gè)標(biāo)志位之類的。

2, ...



樓主的注釋和標(biāo)題都寫明了 “ 利用FreeBSD的kqueue可以在單進(jìn)程中實(shí)現(xiàn)多定時(shí)器, 同時(shí)避免以上弊端!
如果只是說信號(hào)處理,我就不研究定時(shí)的問題了。

你說的準(zhǔn)因?yàn)槟鉺leep是1,如果是10呢?

我主要在嵌入式上編程,嵌入式系統(tǒng)的定時(shí)器都是互不影響的,現(xiàn)在轉(zhuǎn)到*unix下,對(duì)于信號(hào)會(huì)影響sleep等系統(tǒng)調(diào)用很不習(xí)慣。
而且一個(gè)進(jìn)程只有一個(gè)ITIMER_REAL定時(shí)器,我是想能在*nix下找到一個(gè)比較好的解決方法。
您需要登錄后才可以回帖 登錄 | 注冊(cè)

本版積分規(guī)則 發(fā)表回復(fù)

  

北京盛拓優(yōu)訊信息技術(shù)有限公司. 版權(quán)所有 京ICP備16024965號(hào)-6 北京市公安局海淀分局網(wǎng)監(jiān)中心備案編號(hào):11010802020122 niuxiaotong@pcpop.com 17352615567
未成年舉報(bào)專區(qū)
中國互聯(lián)網(wǎng)協(xié)會(huì)會(huì)員  聯(lián)系我們:huangweiwei@itpub.net
感謝所有關(guān)心和支持過ChinaUnix的朋友們 轉(zhuǎn)載本站內(nèi)容請(qǐng)注明原作者名及出處

清除 Cookies - ChinaUnix - Archiver - WAP - TOP