[Linux] signal()函數解析

無心中看到了 signal() 函數,感受對這個函數的原型有諸多疑惑,學習一下,順便分享。函數

1. 原型
Unix系統提供了signal()和sigaction()兩個函數來改變對於信號的處理方法。其中signal()是一個基於sigaction()系統調用的glibc庫函數,其行爲在不一樣的Unix操做系統存在差別,對於可移植性有要求的程序不能使用signal()函數。學習

其原型以下:測試

#include<signal.h>

void ( *signal(int sig, void (*handler)(int)) )(int);

signal函數的第一個參數sig,要傳入但願修改處理函數的信號編號,第二個參數,是一個無返回值、接受一個int形參的函數指針,指向對sig信號的新處理函數。函數的第二個參數能夠有三種選擇,第一種是咱們本身定義的信號處理函數,第二種是傳入SIG_DFL 表示將以前signal所改變的信號處理方式還原,第三種是傳入SIG_IGN,表示處理方式爲忽略該信號,內核會直接將信號丟棄,不會傳遞到進程。操作系統

signal函數的返回值一樣是一個無返回值、接受一個int形參的函數指針。當signal()函數成功執行時,返回值是執行signal函數以前的sig信號的處理函數的指針,若是失敗,則返回SIG_ERR。指針

對信號處理函數指針作這樣的定義,可能便於理解:code

//將sighandler_t定義爲返回值爲void,接受一個int形參的函數的指針
typedef void (*sighandler_t)(int);
//如今咱們能夠這樣書寫signal函數
sighandler_t signal(int sig, sighandler_t handler);

其實glibc裏已經有了如上的定義,只須要咱們定義_GUN_SOURCE特性測試宏,就可使用sighandler_t類型。
2.實例
如下用法用於暫時性修改某信號的信號處理函數,在使用完以後,將信號處理函數重置爲其原來版本。進程

void newhandler(int sig){
    //code to handle the sig
}
void (*prevhandler)(int); //存儲修改以前的信號處理函數
prehandler = signal(SIGINT, newhandler);
if(prehandler == SIG_ERR)  
    cerr<<"Error in signal"<<endl;
    
/*如今,對於中斷信號的處理已經修改成咱們定義的newhandler函數*/

if(signal(SIGINT, prevhandler) == SIG_ERR)
    cerr<<"Error in signal"<<endl;
相關文章
相關標籤/搜索