一個最簡單的信號信號處理例子:
#include <signal.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
void signal_rec(int signum, siginfo_t *info, void *myop)
{
?printf("receive signal %d\n", signum);
?sleep(5);?
}
int main(int argc,char **argv)
{
?struct sigaction act;
?int sig,i;
?
?sigemptyset(&act.sa_mask);
?act.sa_flags = SA_SIGINFO;
?act.sa_sigaction = signal_rec;
?
?for(i=1; i<argc; i++){ //argv[0]為調用程序名
??sig = atoi(argv[i]);
??if(sigaction(sig, &act, NULL) <0){
???printf("install signal error\n");
??}?
?}
?
?while(1){
??sleep(2);
??printf("wait for the signal\n");
?}
}
用法:
將文件保存為test.c,執行make test,編譯生成test文件。
./test 36 39&
命令的意思是注冊信號36,39。命令結果會返回pid,假設為2625。
通過另外一個終端向2625進程發送信號。例如:kill -s 36 2625
則我們可以看到signal_rec函數被執行了。
當我們發送40信號,我們會發送程序終止了,這是因為對于實時信號,默認的操作是終止。
關于信號:
?Linux
支持的信號分為可靠信號和不可靠信號。此處的可靠與不可靠不是指系統是不是可靠的可靠,而是對于可靠信息來說,每一次信號發實都每被加到信號鏈中,所以是
可靠的;而對于不可靠信號來說,如果進程之前已經接收到該信號,則不會被加到信號鏈中,因此對于此次發送的信號來說,對于該進程來說是不知道的,所就是說
信號丟失了,因此是不可靠的。
?不可靠信號主要是在早期信號機制上的信號。一般來說,信號值小于SIGRTMIN的信號為不可靠信號,不可靠信號又稱為非實時信號。可靠信號又稱為實時信號。實時信號是SIGRTMIN和SIGRTMAX間的所有信號。
?我們可以通過kill -l查看SIGRTMIN和SIGRTMAX的值。在Debian系統和Redhat上面,一般SIGRTMIN=33,SIGRTMAX=64。
?Linux既支持新的信號安裝函數sigation以及信號發送函數sigqueue,又支持早期的signal信號安裝函數和kill信號發送函數。
?信號的可靠與不可靠只與信號值有關,與信號的發送和安裝函數無關,也就是說在Linux下即使使用sigaction和sigqueue也不可能將不可靠信號變為可靠信號。
信號發送函數主要有以下幾個:kill,raise,sigqueue,alarm,settimer,abort。
信號安裝函數主要有signal和sigaction。sigaction主要用于與sigqueue系統調用配合使用,主要用于實時信號處理。
信號集操作主要有以下幾個函數:sigemptyset,sigfillset,sigaddset,sigdelset,sigismember。
信號的阻塞的未決主要有以下幾個函數:sigprocmask,sigsuspend,sigpending。
最后以一個讀串口設備時用到的信號處理作為結尾(在上次的初始化串口設備中已經出現過):
在讀取設備數據或者進行網絡應用的時候,為了防止程序進入死鎖,我們需要設置超時操作,即比如我們讀串口設備,嘗試一定時間后仍然沒有響應,則可能設備沒有正常工作。那么在超時以后我們需要退出,否則程序就鎖住了。
我們現在使用SIGALRM信號來進行這個處理(這種方式并不是最好的辦法):
我們假設有一個標記是否超時的全局變量caught_alrm,默認為0,為1時則表示超時。
static volatile sig_atomic_t caught_alrm;
static void sig_alrm(int signo){//信號處理函數,設置超時全局變量為1。
??????? caught_alrm = 1;
??????? return;
}
int a_function{
?……
??? if(signal(SIGALRM, sig_alrm) == SIG_ERR){
??? ?syslog(LOG_ERR,"signal error in function:%s",__FUNCTION__);
??????? return -1;
??? }
??? caught_alrm = 0;
??? alarm(expalarm);
??? do{
??? ?//do something
??? ?//if the work is finished,call alarm(0) to clean the timer.
??? }while(caught_alrm == 0);
}
注:
1、本文為整理以前的工作筆記,如果您要轉載,請注意來源為爾雅,作者覃士國。如果您有任何問題歡迎交流:shiguo.qin@gmail.com
2、
此文僅僅是一則筆記,如果您正在尋找關于Linux信號編程方面的資料,可以參考鄭彥興的一篇文章:http://www-
128.ibm.com/developerworks/cn/linux/l-ipc/part2/index1.html。此文詳細描述了很多關于信
號處理中的結構等內容,是學習信號編程的一篇不錯的教程式文章。
?