(原創) 如何每间格一段时间就执行function? (setitimer()) (C/C++) (C) (OS) (Linux)
setitimer()为Linux的API,并非C语言的Standard Library,setitimer()有两个功能,一是指定一段时间后,才执行某个function,二是每间格一段时间就执行某个function,以下程序demo如何使用setitimer()。
2(C) OOMusou 2006 http://oomusou.cnblogs.com
3
4Filename : timer.cpp
5Compiler : gcc 4.1.0 on Fedora Core 5
6Description : setitimer() set the interval to run function
7Synopsis : #include <sys/time.h>
8 int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue);
9 struct itimerval {
10 struct timerval it_interval;
11 struct timerval it_value;
12 };
13
14 struct timeval {
15 long tv_sec;
16 long tv_usec;
17 }
18Release : 11/25/2006
19*/
20#include <stdio.h> // for printf()
21#include <unistd.h> // for pause()
22#include <signal.h> // for signal()
23#include <string.h> // for memset()
24#include <sys/time.h> // struct itimeral. setitimer()
25
26void printMsg(int);
27
28int main() {
29 // Get system call result to determine successful or failed
30 int res = 0;
31 // Register printMsg to SIGALRM
32 signal(SIGALRM, printMsg);
33
34 struct itimerval tick;
35 // Initialize struct
36 memset(&tick, 0, sizeof(tick));
37 // Timeout to run function first time
38 tick.it_value.tv_sec = 1; // sec
39 tick.it_value.tv_usec = 0; // micro sec.
40 // Interval time to run function
41 tick.it_interval.tv_sec = 1;
42 tick.it_interval.tv_usec = 0;
43 // Set timer, ITIMER_REAL : real-time to decrease timer,
44 // send SIGALRM when timeout
45 res = setitimer(ITIMER_REAL, &tick, NULL);
46 if (res) {
47 printf("Set timer failed!!\n");
48 }
49
50 // Always sleep to catch SIGALRM signal
51 while(1) {
52 pause();
53 }
54
55 return 0;
56}
57
58void printMsg(int num) {
59 printf("%s","Hello World!!\n");
60}
当setitimer()所执行的timer时间到了,会呼叫SIGALRM signal,所以在第30行用signal()将要执行的function指定给SIGALRM。
在第43行呼叫setitimer()设定timer,但setitimer()第二个参数是sturct,负责设定timeout时间,所以第36行到第40行设定此struct。itimerval.it_value设定第一次执行function所延迟的秒数,itimerval.it_interval设定以后每几秒执行function,所以若只想延迟一段时间执行function,只要设定itimerval.it_value即可,若要设定间格一段时间就执行function,则it_value和it_interval都要设定,否则funtion的第一次无法执行,就别说以后的间隔执行了。
第36行和第39行的tv_sec为sec,第37行和40行为micro sec(0.001 sec)。
第43行的第一个参数ITIMER_REAL,表示以real-time方式减少timer,在timeout时会送出SIGALRM signal。第三个参数会存放旧的timeout值,如果不需要的话,指定NULL即可。
第47行的pause(),命令系统进入sleep状态,等待任何signal,一定要用while(1)无穷循环执行pause(),如此才能一直接收SIGALRM signal以间隔执行function,若拿掉while(1),则function只会执行一次而已。
在JavaScript设定timer,只要一行就好,没想到在Linux下这么麻烦。
See Also
如何每间格一段时间就执行function? (setInterval())(Web) (JavaScript)
Reference
Linux C函式库详解辞典 P.396 pause(), 徐千洋 着, 旗标出版社
C程序设计 500个应用范例技巧大全集 P.703 ~ P.706, 平田 丰 着/ 黄政凯 译, 博硕文化