[ 永遠的UNIX::UNIX技術資料的寶庫 ]   GB | BIG5

首頁 > 編程技術 > C/C++ > 正文
在C程式中要怎用 sleep() 才能夠sleep小一秒?
Steve Hayman (2001-08-22 16:44:23)
  首先要注意的是,你只能指定 delay 的「最短」時間;實際上會 delay 多久和 
  系統的 scheduling 方式有關,例如系統當時有負載。如果你倒楣的話,它還可 
  能會 delay 蠻長的時間。 

  並沒有一個標準函式能夠在「小睡」(很短的 sleep)期間提供你計數的功能。 
  某些系統有提供 usleep(n) 的函式,它能夠暫停執行 n 微秒(microsecond) 
  的時間。如果你所使用的系統沒有提供 usleep() 函式,那以下有可在 BSD, 
  System V 使用中的作法。 

  接下來的這段程式碼是 Doug Gwyn 在 System V 中模擬 4BSD 並利用 4BSD 
  中的 select() 系統呼叫。Doung 自己都叫它為 'nap()' ;你也可以把它叫做 
  "usleep()"; 

  /* 
      usleep -- support routine for 4.2BSD system call emulations 
      last edit:  29-Oct-1984     D A Gwyn 
  */ 

extern int        select(); 

int 
usleep( usec )                            /* returns 0 if ok, else -1 */ 
      long                usec;           /* delay in microseconds */ 
      { 
      static struct                       /* `timeval' */ 
              { 
              long        tv_sec;         /* seconds */ 
              long        tv_usec;        /* microsecs */ 
              }   delay;          /* _select() timeout */ 

      delay.tv_sec = usec / 1000000L; 
      delay.tv_usec = usec % 1000000L; 

      return select( 0, (long *)0, (long *)0, (long *)0, &delay ); 
      } 

On System V you might do it this way: 

/* 
subseconds sleeps for System V - or anything that has poll() 
Don Libes, 4/1/1991 

The BSD analog to this function is defined in terms of 
microseconds while poll() is defined in terms of milliseconds. 
For compatibility, this function provides accuracy "over the long 
run" by truncating actual requests to milliseconds and 
accumulating microseconds across calls with the idea that you are 
probably calling it in a tight loop, and that over the long run, 
the error will even out. 

If you aren't calling it in a tight loop, then you almost 
certainly aren't making microsecond-resolution requests anyway, 
in which case you don't care about microseconds.  And if you did, 
you wouldn't be using UNIX anyway because random system 
indigestion (i.e., scheduling) can make mincemeat out of any 
timing code. 

Returns 0 if successful timeout, -1 if unsuccessful. 

*/ 

#include <poll.h> 

int 
usleep(usec) 
unsigned int usec;                /* microseconds */ 
{ 
      static subtotal = 0;        /* microseconds */ 
      int msec;                   /* milliseconds */ 

      /* 'foo' is only here because some versions of 5.3 have 
       * a bug where the first argument to poll() is checked 
       * for a valid memory address even if the second argument is 0. 
       */ 
      struct pollfd foo; 

      subtotal += usec; 
      /* if less then 1 msec request, do nothing but remember it */ 
      if (subtotal < 1000) return(0); 
      msec = subtotal/1000; 
      subtotal = subtotal%1000; 
      return poll(&foo,(unsigned long)0,msec); 
} 

  在 System V 或其他 非-BSD 的 Unix 中要使用這類的「小睡」程式,可以用 
  Jon Zeeff 的 s5nap,它曾被發表在 comp.sources.misc, volume 4 中。它 
  需要安裝一個驅動程式,但是裝好就可以跑得很好。(它的精確度會受到 
  kernel 中 HZ 這個變數的影響,因為它是用到了 kernel 中的 delay() 函 
  式。) 

  現在很多較新版本的 Unix 都有提供這類的「小睡」功能了。  (http://www.fanqiang.com)
    進入【UNIX論壇

相關文章

======
 

★  樊強制作 歡迎分享  ★