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

Chinaunix

標(biāo)題: 利用kqueue在單進(jìn)程中實(shí)現(xiàn)多定時(shí)器 [打印本頁(yè)]

作者: doctorjxd    時(shí)間: 2007-11-16 22:31
標(biāo)題: 利用kqueue在單進(jìn)程中實(shí)現(xiàn)多定時(shí)器
利用C語(yǔ)言函數(shù),一般在一個(gè)進(jìn)程中只能有一個(gè)定時(shí)器。要想實(shí)現(xiàn)多定時(shí)器,一般是在這個(gè)定時(shí)器中使用累計(jì)計(jì)數(shù)來(lái)實(shí)現(xiàn)。當(dāng)定時(shí)器比較多,和定時(shí)間隔差別很大,如0.02秒和10秒,計(jì)數(shù)就帶來(lái)了很大的額外開(kāi)銷。
  當(dāng)然也可用循環(huán)嵌入sleep,nanosleep的方法,但是這增加了循環(huán)內(nèi)不必要的阻塞。
  多進(jìn)程和多線程的方法,增加了程序的復(fù)雜度和開(kāi)銷。

  利用FreeBSD的kqueue可以在單進(jìn)程中實(shí)現(xiàn)多定時(shí)器, 同時(shí)避免以上弊端。
  下面是我寫(xiě)的一小段代碼,與大家共享,獻(xiàn)丑了,呵呵。轉(zhuǎn)載請(qǐng)標(biāo)明來(lái)自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 編輯 ]
作者: benjiam    時(shí)間: 2007-11-19 09:14
不錯(cuò)! 有無(wú)kqueue 更深入的文檔?
作者: doctorjxd    時(shí)間: 2007-11-19 11:26
原帖由 benjiam 于 2007-11-19 09:14 發(fā)表
不錯(cuò)! 有無(wú)kqueue 更深入的文檔?


http://wiki.netbsd.se/kqueue_tutorial

要在freebsd上編譯kqueue的例子 要加頭文件 <sys/types.h>
作者: jameszxj    時(shí)間: 2007-12-29 09:15
標(biāo)題: 討論一下
這樣的定時(shí)能準(zhǔn)確嗎?我在FreeBSD6.3上測(cè)試好像不太準(zhǔn),

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

加了個(gè)sleep(10)試了。
作者: googhu    時(shí)間: 2007-12-29 15:18
用epoll能實(shí)現(xiàn)這個(gè)嗎?怎樣實(shí)現(xiàn)?
作者: lgfang    時(shí)間: 2007-12-29 15:19
原帖由 jameszxj 于 2007-12-29 09:15 發(fā)表
這樣的定時(shí)能準(zhǔn)確嗎?我在FreeBSD6.3上測(cè)試好像不太準(zhǔn),

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


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

2,即使你做其它事情的時(shí)間很長(zhǎng)也不會(huì)影響定時(shí)的準(zhǔn)確性(下面的程序測(cè)試為證),只不過(guò)有可能你自己來(lái)不及及時(shí)響應(yīng)。但正如我在第一點(diǎn)里指出的,那是你自己的問(wè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 編輯 ]
作者: lgfang    時(shí)間: 2007-12-29 20:02
http://people.freebsd.org/~jlemon/papers/kqueue.pdf
作者: googhu    時(shí)間: 2007-12-29 21:18
epoll好像不能實(shí)現(xiàn)
EV_SET(&changes[1],1,EVFILT_TIMER, EV_ADD | EV_ENABLE, 0, 2000, 0);
epoll沒(méi)有EVFILT_TIMER這個(gè)參數(shù),不一樣~~
作者: jameszxj    時(shí)間: 2007-12-31 19:26
原帖由 lgfang 于 2007-12-29 15:19 發(fā)表


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

2, ...



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

你說(shuō)的準(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è)比較好的解決方法。




歡迎光臨 Chinaunix (http://www.72891.cn/) Powered by Discuz! X3.2