多線(xiàn)程中定時(shí)器的使用
unsigned int alarm(unsigned int seconds);
這個(gè)函數在使用上很簡(jiǎn)單,第一次調用這個(gè)函數的時(shí)候是設置定時(shí)器的初值,下一次調用是重新設置這個(gè)值,并會(huì )返回上一次定時(shí)的剩余時(shí)間。
#i nclude
int setitimer(int which, const struct itimerval *value,
這個(gè)函數使用起來(lái)稍微有點(diǎn)說(shuō)法,首先是第一個(gè)參數which的值,這個(gè)參數設置timer的計時(shí)策略,which有三種狀態(tài)分別是:
ITIMER_REAL:使用系統時(shí)間來(lái)計數,時(shí)間為0時(shí)發(fā)出SIGALRM信號,這種定時(shí)能夠得到一個(gè)精準的定時(shí),當然這個(gè)定時(shí)是相對的,因為到了微秒級別我們的處理器本身就不夠精確。
ITIMER_VIRTUAL:使用進(jìn)程時(shí)間也就是進(jìn)程分配到的時(shí)間片的時(shí)間來(lái)計數,時(shí)間為0是發(fā)出SIGVTALRM信號,這種定時(shí)顯然不夠準確,因為系統給進(jìn)程分配時(shí)間片不由我們控制。
ITIMER_PROF:上面兩種情況都能夠觸發(fā)
第二個(gè)參數參數value涉及到兩個(gè)結構體:
struct itimerval {
struct timeval {
};
在結構體itimerval中it_value是定時(shí)器當前的值,it_interval是當it_value的為0后重新填充的值。而timeval結構體中的兩個(gè)變量就簡(jiǎn)單了一個(gè)是秒一個(gè)是微秒。
上面是這兩個(gè)定時(shí)函數的說(shuō)明,這個(gè)函數使用本不是很難,可以說(shuō)是很簡(jiǎn)單,但是碰到具體的應用的時(shí)候可能就遇到問(wèn)題了,在多進(jìn)程編程中使用一般不會(huì )碰到什么問(wèn)題,這里說(shuō)的這些問(wèn)題主要體現在多線(xiàn)程編程中。比如下面這個(gè)程序
#i nclude
#i nclude
#i nclude
#i nclude
#i nclude
#i nclude
void sig_handler(int signo)
{
}
void *pthread_func()
{
}
int main(int argc, char **argv)
{
}
這個(gè)程序的理想結果是:
main thread
alarm signal
alarm signal
alarm signal
alarm signal
alarm signal
main thread
可事實(shí)上并不是這樣的,它的結果是:
main pthread
alarm signal
main pthread
alarm signal
main pthread
為什么會(huì )出現這種情況呢?是因為發(fā)送給工作線(xiàn)程的信號中斷的主線(xiàn)程的sleep,并且這個(gè)中情況只影響主線(xiàn)程而不會(huì )影響到其他的工作線(xiàn)程。我們怎么才能解決這種問(wèn)題呢,最簡(jiǎn)單的方法是修改這個(gè)程序,修改這個(gè)線(xiàn)程主線(xiàn)程使用alarm,工作線(xiàn)程使用sleep。這樣就能夠達到我們的要求,但是有時(shí)候有不能簡(jiǎn)單的這樣操作。所以我們就需要進(jìn)一步的修改我們的程序。在這里我第一個(gè)想到的是使用signal(SIGALRM, SIG_IGN),可是這個(gè)是設置整個(gè)進(jìn)程對這個(gè)信號的響應方式,經(jīng)過(guò)測試也確實(shí)不能完成我期望的功能,那么怎么辦呢?有這樣一個(gè)函數pthread_sigmask,線(xiàn)程中的信號屏蔽,函數的原型及相關(guān)函數為:
#i nclude
int pthread_sigmask(int how, const sigset_t *restrict set, sigset_t *restrict oset);
函數中第一個(gè)參數how有三個(gè)值SIG_BLOCK、SIG_SETMASK和SIG_UNBLOCK這里我們是用第二個(gè)值SIG_SETMASK
int sigemptyset(sigset_t *set);
int sigaddset(sigset_t *set, int signum);
然后我們改造我們的程序為:
#i nclude
#i nclude
#i nclude
#i nclude
#i nclude
#i nclude
void sig_handler(int signo)
{
}
void *pthread_func()
{
}
int main(int argc, char **argv)
{
}
這個(gè)時(shí)候我們就能夠看到我們想要的結果了。
這里再附一個(gè)setitimer的使用范例
#i nclude
#i nclude
#i nclude
#i nclude
#i nclude
#i nclude
struct itimerval timerval;
void sig_handler(int signo)
{
}
void *pthread_func()
{
}
int main(int argc, char **argv)
{
}
評論