linux使用select實(shí)現(xiàn)精確定時(shí)器詳解
更新時(shí)間:2013年11月14日 11:11:11 作者:
本文講述如何使用select實(shí)現(xiàn)超級(jí)時(shí)鐘。使用select函數(shù),我們能實(shí)現(xiàn)微妙級(jí)別精度的定時(shí)器。同時(shí),select函數(shù)也是我們?cè)诰帉懛亲枞绦驎r(shí)經(jīng)常用到的一個(gè)函數(shù)
在編寫程序時(shí),我們經(jīng)常會(huì)用到定時(shí)器。首先看看select函數(shù)原型如下:
int select(int nfds, fd_set *readfds, fd_set *writefds,
fd_set *exceptfds, struct timeval *timeout);
參數(shù)說(shuō)明:
slect的第一個(gè)參數(shù)nfds為fdset集合中最大描述符值加1,fdset是一個(gè)位數(shù)組,其大小限制為_(kāi)_FD_SETSIZE(1024),位數(shù)組的每一位代表其對(duì)應(yīng)的描述符是否需要被檢查。
select的第二三四個(gè)參數(shù)表示需要關(guān)注讀、寫、錯(cuò)誤事件的文件描述符位數(shù)組,這些參數(shù)既是輸入?yún)?shù)也是輸出參數(shù),可能會(huì)被內(nèi)核修改用于標(biāo)示哪些描述符上發(fā)生了關(guān)注的事件。所以每次調(diào)用select前都需重新初始化fdset。
timeout參數(shù)為超時(shí)時(shí)間,該結(jié)構(gòu)會(huì)被內(nèi)核修改,其值為超時(shí)剩余的時(shí)間。
利用select實(shí)現(xiàn)定時(shí)器,需要利用其timeout參數(shù),注意到:
1)select函數(shù)使用了一個(gè)結(jié)構(gòu)體timeval作為其參數(shù)。
2)select函數(shù)會(huì)更新timeval的值,timeval保持的值為剩余時(shí)間。
如果我們指定了參數(shù)timeval的值,而將其他參數(shù)都置為0或者NULL,那么在時(shí)間耗盡后,select函數(shù)便返回,基于這一點(diǎn),我們可以利用select實(shí)現(xiàn)精確定時(shí)。
timeval的結(jié)構(gòu)如下:
struct timeval{
long tv_sec;/*secons*
long tv_usec;/*microseconds*/
}
我們可以看出其精確到microseconds也即微妙。
一、秒級(jí)定時(shí)器
void seconds_sleep(unsigned seconds){
struct timeval tv;
tv.tv_sec=seconds;
tv.tv_usec=0;
int err;
do{
err=select(0,NULL,NULL,NULL,&tv);
}while(err<0 && errno==EINTR);
}
二、毫秒級(jí)別定時(shí)器
void milliseconds_sleep(unsigned long mSec){
struct timeval tv;
tv.tv_sec=mSec/1000;
tv.tv_usec=(mSec%1000)*1000;
int err;
do{
err=select(0,NULL,NULL,NULL,&tv);
}while(err<0 && errno==EINTR);
}
三、微妙級(jí)別定時(shí)器
void microseconds_sleep(unsigned long uSec){
struct timeval tv;
tv.tv_sec=uSec/1000000;
tv.tv_usec=uSec%1000000;
int err;
do{
err=select(0,NULL,NULL,NULL,&tv);
}while(err<0 && errno==EINTR);
}
現(xiàn)在我們來(lái)編寫幾行代碼看看定時(shí)效果吧。
#include <stdio.h>
#include <sys/time.h>
#include <errno.h>
int main()
{
int i;
for(i=0;i<5;++i){
printf("%d\n",i);
//seconds_sleep(1);
//milliseconds_sleep(1500);
microseconds_sleep(1900000);
}
}
注:timeval結(jié)構(gòu)體中雖然指定了一個(gè)微妙級(jí)別的分辨率,但內(nèi)核支持的分別率往往沒(méi)有這么高,很多unix內(nèi)核將超時(shí)值向上舍入成10ms的倍數(shù)。此外,加上內(nèi)核調(diào)度延時(shí)現(xiàn)象,即定時(shí)器時(shí)間到后,內(nèi)核還需要花一定時(shí)間調(diào)度相應(yīng)進(jìn)程的運(yùn)行。因此,定時(shí)器的精度,最終還是由內(nèi)核支持的分別率決定。
分類: Linux
復(fù)制代碼 代碼如下:
int select(int nfds, fd_set *readfds, fd_set *writefds,
fd_set *exceptfds, struct timeval *timeout);
參數(shù)說(shuō)明:
slect的第一個(gè)參數(shù)nfds為fdset集合中最大描述符值加1,fdset是一個(gè)位數(shù)組,其大小限制為_(kāi)_FD_SETSIZE(1024),位數(shù)組的每一位代表其對(duì)應(yīng)的描述符是否需要被檢查。
select的第二三四個(gè)參數(shù)表示需要關(guān)注讀、寫、錯(cuò)誤事件的文件描述符位數(shù)組,這些參數(shù)既是輸入?yún)?shù)也是輸出參數(shù),可能會(huì)被內(nèi)核修改用于標(biāo)示哪些描述符上發(fā)生了關(guān)注的事件。所以每次調(diào)用select前都需重新初始化fdset。
timeout參數(shù)為超時(shí)時(shí)間,該結(jié)構(gòu)會(huì)被內(nèi)核修改,其值為超時(shí)剩余的時(shí)間。
利用select實(shí)現(xiàn)定時(shí)器,需要利用其timeout參數(shù),注意到:
1)select函數(shù)使用了一個(gè)結(jié)構(gòu)體timeval作為其參數(shù)。
2)select函數(shù)會(huì)更新timeval的值,timeval保持的值為剩余時(shí)間。
如果我們指定了參數(shù)timeval的值,而將其他參數(shù)都置為0或者NULL,那么在時(shí)間耗盡后,select函數(shù)便返回,基于這一點(diǎn),我們可以利用select實(shí)現(xiàn)精確定時(shí)。
timeval的結(jié)構(gòu)如下:
復(fù)制代碼 代碼如下:
struct timeval{
long tv_sec;/*secons*
long tv_usec;/*microseconds*/
}
我們可以看出其精確到microseconds也即微妙。
一、秒級(jí)定時(shí)器
復(fù)制代碼 代碼如下:
void seconds_sleep(unsigned seconds){
struct timeval tv;
tv.tv_sec=seconds;
tv.tv_usec=0;
int err;
do{
err=select(0,NULL,NULL,NULL,&tv);
}while(err<0 && errno==EINTR);
}
二、毫秒級(jí)別定時(shí)器
復(fù)制代碼 代碼如下:
void milliseconds_sleep(unsigned long mSec){
struct timeval tv;
tv.tv_sec=mSec/1000;
tv.tv_usec=(mSec%1000)*1000;
int err;
do{
err=select(0,NULL,NULL,NULL,&tv);
}while(err<0 && errno==EINTR);
}
三、微妙級(jí)別定時(shí)器
復(fù)制代碼 代碼如下:
void microseconds_sleep(unsigned long uSec){
struct timeval tv;
tv.tv_sec=uSec/1000000;
tv.tv_usec=uSec%1000000;
int err;
do{
err=select(0,NULL,NULL,NULL,&tv);
}while(err<0 && errno==EINTR);
}
現(xiàn)在我們來(lái)編寫幾行代碼看看定時(shí)效果吧。
復(fù)制代碼 代碼如下:
#include <stdio.h>
#include <sys/time.h>
#include <errno.h>
int main()
{
int i;
for(i=0;i<5;++i){
printf("%d\n",i);
//seconds_sleep(1);
//milliseconds_sleep(1500);
microseconds_sleep(1900000);
}
}
注:timeval結(jié)構(gòu)體中雖然指定了一個(gè)微妙級(jí)別的分辨率,但內(nèi)核支持的分別率往往沒(méi)有這么高,很多unix內(nèi)核將超時(shí)值向上舍入成10ms的倍數(shù)。此外,加上內(nèi)核調(diào)度延時(shí)現(xiàn)象,即定時(shí)器時(shí)間到后,內(nèi)核還需要花一定時(shí)間調(diào)度相應(yīng)進(jìn)程的運(yùn)行。因此,定時(shí)器的精度,最終還是由內(nèi)核支持的分別率決定。
分類: Linux
相關(guān)文章
監(jiān)控服務(wù)器swap并重啟php的Shell腳本
有一臺(tái)服務(wù)器老是交換扇區(qū)占滿然后失去響應(yīng),很煩,因?yàn)榧虞d了以前別人寫的一個(gè)php擴(kuò)展,效率低,資源占用大,悲劇的是現(xiàn)在還沒(méi)有可以替換的東西2014-03-03
Shell實(shí)現(xiàn)判斷進(jìn)程是否存在并重新啟動(dòng)腳本分享
這篇文章主要介紹了Shell實(shí)現(xiàn)判斷進(jìn)程是否存在并重新啟動(dòng)腳本分享,本文給出了兩個(gè)實(shí)現(xiàn)腳本,分簡(jiǎn)潔版和詳細(xì)版,需要的朋友可以參考下2014-09-09
Bash?Shell中單引號(hào)和雙引號(hào)的區(qū)別小結(jié)
在Bash中,單引號(hào)和雙引號(hào)都能定義字符串,但它們處理變量擴(kuò)展、特殊字符的方式不同,本文就來(lái)介紹一下Bash?Shell中單引號(hào)和雙引號(hào)的區(qū)別小結(jié),感興趣的可以了解一下2024-11-11
Linux中執(zhí)行shell腳本的4種方法總結(jié)
這篇文章主要介紹了Linux中執(zhí)行shell腳本的4種方法總結(jié),即在Linux中運(yùn)行shell腳本的4種方法,需要的朋友可以參考下2014-08-08
Linux shell編程中IO和條件及循環(huán)處理的細(xì)節(jié)問(wèn)題討論
這篇文章主要介紹了Linux shell編程中IO和條件及循環(huán)處理的細(xì)節(jié)問(wèn)題討論,需要的朋友可以參考下2016-02-02

