09 5月 2008

[Linux] 基本的Timer介紹

Linux提供了兩種基本的Timer機制可以使用:
  • alarm
  • setitimer

● alarm
#include <unistd.h>
unsigned int alarm(unsigned int seconds);
這是一個簡單的設定Timer介面。當呼叫了alarm( n )後,等待n秒後,就會觸發一次的SIGALRM的signal,所以必須要在呼叫alarm前,先設好SIGALRM的handler function才行。而當乎呼alarm(0)時,則表示停止當前的timer處理,不要發出SIGALRM的signal。

Return value : 返回上一次呼叫alarm的剩餘秒數。若未設定alarm,則返回0。

Example : 第一次等待1秒後觸發Timer,之後每隔2秒觸發一次。
#include <iostream>
#include <unistd.h>
#include <signal.h>

using namespace std;

void my_alarm_handler(int a)
{
cerr<<"my_alarm_handler"<<endl;
alarm(2);//重新設定
}

int main()
{
signal( SIGALRM, my_alarm_handler );
alarm(1);

while(1){}

return 0;
}

--------------------------------------------------------------------------------------
● setitimer
#include <sys/time.h>
#define ITIMER_REAL 0
#define ITIMER_VIRTUAL 1
#define ITIMER_PROF 2

int getitimer(int which, struct itimerval *value);

int setitimer(int which, const struct itimerval *value,
struct itimerval *ovalue);
setitimer與getitimer提供了三種類別的Timer使用:
  • ITIMER_REAL : 以系統真實的時間來計算,觸發時會送出SIGALRM
  • ITIMER_VIRTUAL : 只計算process真正在執行的時間(在User Mode的處理),觸發時會送出SIGVTALRM
  • ITIMER_PROF : 計算該process在User Mode與Kernel Mode的處理時間,觸發時送出SIGPROF
透過第一個參數which指定要使用哪一種Timer (ITIMER_REAL, ITIMER_VIRTUAL, ITIMER_PROF )。setitimer是用來設定該種Timer的觸發時間為多少。getitimer則是取得上一次Timer設定的時間。設定的內容是一個系統內建的struct itimerval:
struct itimerval {
   struct timeval it_interval; /* next value : 下一次觸發所需的時間*/
   struct timeval it_value; /* current value : 目前距離觸發時間點 剩餘的時間*/
};
struct timeval {
   long tv_sec; /* seconds */
   long tv_usec; /* microseconds */
};

setitimer由第二個參數value設定觸發的時間。第三個參數ovalue用來取得上一次 setitimer設定的itimerval值(此參數可以為NULL)。值得注意的是,根據itimerval裡變數的意義,當it_interval設定為0時,Timer只會觸發一次。而it_value設定為0時,代表Timer結束。

Return value : 如果成功則return 0,失敗則return -1。

Example : 第一次等待1秒後觸發Timer,之後每隔2秒觸發一次。
#include <iostream>
#include <sys/time.h>
#include <signal.h>

using namespace std;

void my_alarm_handler(int a)
{
cerr<<"test "<<endl;
}


int main(){
struct itimerval t;
t.it_interval.tv_usec = 0;
t.it_interval.tv_sec = 2;
t.it_value.tv_usec = 0;
t.it_value.tv_sec = 1;

if( setitimer( ITIMER_REAL, &t, NULL) < 0 ){
cerr<<"settimer error."<<endl;
return -1;
}
signal( SIGALRM, my_alarm_handler );

while(1){
sleep(2);
}
return 0;
}

● 根據以上,可知Linux內建的Timer還是有點簡陋,而且setitimer同一時間只能處理3個Timer,如果應用程式需要多個Timer的話,這個Linux內建的Timer可能就不敷需求了!

11 則留言:

匿名 提到...

那請問有什麼方式可訂3個以上的Timer呢
謝謝

Falldog 提到...

若要用到多個Timer,可以試試GTK提供的Timer函式:g_timeout_add(),可以參考這篇教學

匿名 提到...

想不到您回覆如此快,
到GTK似乎是用在XWindow, 若是在開發server programing要如何使用呢?
謝謝

Falldog 提到...

因為一有留言,就有Email通知我XD

你可以不要用到GTK的Window介面,只要用到它的Timer部分就好啦:)

這是我之前測試的Code,給你參考一下:
#include <iostream>
#include <glib.h>

using namespace std;

gboolean test(gpointer data)
{
cerr<<"test "<<(char*)data<<endl;
return TRUE;//True : Timer繼續, False : Timer結束
}


int main(){
char * a = "100";
char * b = "200";
g_timeout_add(1000, (GSourceFunc)test, a);
g_timeout_add(2000, (GSourceFunc)test, b);
GMainLoop* my_main = g_main_loop_new(NULL,TRUE);
g_main_loop_run( my_main );
return 0;
}

匿名 提到...

請問一下您的glib.h是放在那, 我的glib.h有兩個版本/usr/include/glib-2.0/glib.h及
/usr/include/glib-1.2/glib.h
但用g++ compile會有問題, 您的OS版本是什麼呢
謝謝

Falldog 提到...

我之前測試的系統是Ubuntu 8.04,記得是安裝完libgtk2.0-dev後,就可以compile成功了。

你在compile時有加上參數:`pkg-config --cflags gtk+-2.0`嗎???

詳細的GTK中文教學 這邊有

匿名 提到...

謝啦...我測試可以了
不過我的參數是: pkg-config --cflags --libs glib-2.0 gthread-2.0
對了, 我覺得這程式若改用pthread來執行timer會較好.
不過還是感謝您給我個tips.

匿名 提到...

不好意思,請教一下,你說同一時間只能處理3個Timer,是指同一時間只能處理三種類別的Timer嗎?還是同一時間處理的3個Timer可以都是ITIMER_REAL這種類型的呢?

Falldog 提到...

我的意思是…同一時間只能處理三種類別的Timer,不是同一時間處理三個相同類別的Timer。

所以同一時間只能處理ITIMER_REAL, ITIMER_VIRTUAL, ITIMER_PROF各一個。

匿名 提到...

您好~ 我是Linux新手,請教一些問題...
我有試了三個GTK的Timer,並分別給於不同的權限,不過若權限低的處理過久,權限高的也不會從中中斷,只有當2個Timer同時醒來時才會先執行權限高的,且最小的時間只能設2ms。
以前都是寫單晶片的程式,需求像是Real Time機制,不知有何其它方式或是可參考的資源...

Falldog 提到...

抱歉,我對這部分沒有研究~@@