Signal ()函式詳細介紹 Linux函式

2puT發表於2016-07-08

signal()函式理解

<signal.h> 這個標頭檔案中。

signal(引數1,引數2);

引數1:我們要進行處理的訊號。系統的訊號我們可以再終端鍵入 kill -l檢視(64)。其實這些訊號時系統定義的巨集。

引數2:我們處理的方式(是系統預設還是忽略還是捕獲)。

一般有3中方式進行操作。

1eg: signal(SIGINT ,SIG_ING );

//SIG_ING 代表忽略SIGINT訊號SIGINT訊號代表由InterruptKey產生,通常是CTRL +C 或者是DELETE 。傳送給所有ForeGround Group的程式。

下面我們寫個死迴圈:


這時我們儲存執行。

按下CTRL _C程式沒有反應。這就對了

如果我們想結束該程式可以按下CTRL +\來結束

其實當我們按下CTRL +\組合鍵時,是產生了SIGQUIT訊號

2eg: signal(SIGINT ,SIG_DFL );

//SIGINT訊號代表由InterruptKey產生,通常是CTRL +C或者是DELETE。傳送給所有ForeGroundGroup的程式。 SIG_DFL代表執行系統預設操作,其實對於大多數訊號的系統預設動作時終止該程式。這與不寫此處理函式是一樣的。

我們將上面的程式改成


這時就可以按下CTRL +C 來終止該程式了。把signal(SIGINT,SIG_DFL);這句去掉,效果是一樣的。

3void ( *signal( int sig, void (* handler)( int )))( int );
int (*p)();
這是一個函式指標, p所指向的函式是一個不帶任何引數, 並且返回值為int的一個函式.
int (*fun())();
這個式子與上面式子的區別在於用fun()代替了p,fun()是一個函式,所以說就可以看成是fun()這個函式執行之後,它的返回值是一個函式指標,這個函式指標(其實就是上面的p)所指向的函式是一個不帶任何引數,並且返回值為int的一個函式.

void (*signal(int signo, void (*handler)(int)))(int);就可以看成是signal()函式(它自己是帶兩個引數,一個為整型,一個為函式指標的函式),而這個signal()函式的返回值也為一個函式指標,這個函式指標指向一個帶一個整型引數,並且返回值為void的一個函式.

在寫訊號處理函式時對於訊號處理的函式也是void sig_fun(int signo);這種型別,恰好與上面signal()函式所返回的函式指標所指向的函式是一樣的.void ( *signal() )( int );

signal是一個函式, 它返回一個函式指標, 後者所指向的函式接受一個整型引數 且沒有返回值, 仔細看, 是不是siganal( int signo, void (*handler)(int) )的第2個引數了,對了,其實他所返回的就是 signal的第2個訊號處理函式,指向訊號處理函式,就可以執行函式了( signal內部時, signal把訊號做為引數傳遞給handler訊號處理函式,接著 signal函式返回指標,並且又指向訊號處理函式,就開始執行它)

那麼,signal函式的引數又是如何呢?signal函式接受兩個引數:一個整型的訊號編號,以及一個指向使用者定義的訊號處理函式的指標。我們此前已經定義了指向使用者定義的訊號處理函式的指標sfp

void (*sfp)(int);

sfp 的型別可以通過將上面的宣告中的sfp去掉而得到,即void (*)(int)。此外,signal函式的返回值是一個指向呼叫前的使用者定義訊號處理函式的指標,這個指標的型別與sfp指標型別一致。因此,我們可以如下宣告signal函式:

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

同樣地,使用typedef可以簡化上面的函式宣告:

typedef void (*HANDLER)(int);
HANDLER signal(int, HANDLER);

Ok;看個例子:


此程式是對當我們按下CTRL +C鍵時,會執行我們定義的訊號處理函式。

每當我們按下CTRL +C鍵時會列印該訊號的number.可以看出該訊號的num2

要想退出可以按下CTRL +\ 列印結果為最後一行。

一些常用的Signal 如下:

注:下面是從百度文庫中找的(*^__^*) 嘻嘻……

Signal Description
SIGABRT 由呼叫abort函式產生,程式非正常退出
SIGALRM 用alarm函式設定的timer超時或setitimer函式設定的interval timer超時
SIGBUS 某種特定的硬體異常,通常由記憶體訪問引起
SIGCANCEL 由Solaris Thread Library內部使用,通常不會使用
SIGCHLD 程式Terminate或Stop的時候,SIGCHLD會傳送給它的父程式。預設情況下該Signal會被忽略
SIGCONT 當被stop的程式恢復執行的時候,自動傳送
SIGEMT 和實現相關的硬體異常
SIGFPE 數學相關的異常,如被0除,浮點溢位,等等
SIGFREEZE Solaris專用,Hiberate或者Suspended時候傳送
SIGHUP 傳送給具有Terminal的Controlling Process,當terminal被disconnect時候傳送
SIGILL 非法指令異常
SIGINFO BSD signal。由Status Key產生,通常是CTRL+T。傳送給所有Foreground Group的程式
SIGINT 由Interrupt Key產生,通常是CTRL+C或者DELETE。傳送給所有ForeGround Group的程式
SIGIO 非同步IO事件
SIGIOT 實現相關的硬體異常,一般對應SIGABRT
SIGKILL 無法處理和忽略。中止某個程式
SIGLWP 由Solaris Thread Libray內部使用
SIGPIPE 在reader中止之後寫Pipe的時候傳送
SIGPOLL 當某個事件傳送給Pollable Device的時候傳送
SIGPROF Setitimer指定的Profiling Interval Timer所產生
SIGPWR 和系統相關。和UPS相關。
SIGQUIT 輸入Quit Key的時候(CTRL+\)傳送給所有Foreground Group的程式
SIGSEGV 非法記憶體訪問
SIGSTKFLT Linux專用,數學協處理器的棧異常
SIGSTOP 中止程式。無法處理和忽略。
SIGSYS 非法系統呼叫
SIGTERM 請求中止程式,kill命令預設傳送
SIGTHAW Solaris專用,從Suspend恢復時候傳送
SIGTRAP 實現相關的硬體異常。一般是除錯異常
SIGTSTP Suspend Key,一般是Ctrl+Z。傳送給所有Foreground Group的程式
SIGTTIN 當Background Group的程式嘗試讀取Terminal的時候傳送
SIGTTOU 當Background Group的程式嘗試寫Terminal的時候傳送
SIGURG 當out-of-band data接收的時候可能傳送
SIGUSR1 使用者自定義signal 1
SIGUSR2 使用者自定義signal 2
SIGVTALRM setitimer函式設定的Virtual Interval Timer超時的時候
SIGWAITING Solaris Thread Library內部實現專用
SIGWINCH 當Terminal的視窗大小改變的時候,傳送給Foreground Group的所有程式
SIGXCPU 當CPU時間限制超時的時候
SIGXFSZ 程式超過檔案大小限制
SIGXRES Solaris專用,程式超過資源限制的時候傳送

相關文章