void(* signal(int sig,void(* func)(int)))(int);
設置處理信號的功能
指定使用sig指定的信號編號處理信號的方法。 參數func指定程序可以處理信號的三種方式之一:
- 默認處理(SIG_DFL):信號由該特定信號的默認動作處理。
- 忽略信號(SIG_IGN):忽略信號,即使沒有意義,代碼執行仍將繼續。
- 函數處理程序:定義一個特定的函數來處理信號。
或 SIG_DFL 要么 SIG_IGN 被設置為程序啟動時每個支持信號的默認信號處理行為。
參數
- SIG
-
設置處理功能的
信號值
。以下宏常量表達式標識標准信號值:
宏 信號 SIGABRT (信號中止)異常終止,例如由...發起 退出 功能。 SIGFPE (信號浮點異常)錯誤的算術運算,例如零分頻或導致溢出的運算(不一定是浮點運算)。 SIGILL (信號非法指令)無效的功能圖像,例如非法指令。這通常是由於代碼中的損壞或嘗試執行數據。 SIGINT (信號中斷)交互式注意信號。通常由應用程序用戶生成。 SIGSEGV (信號分段違規)對存儲的無效訪問:當程序試圖在已分配的內存之外讀取或寫入時。 SIGTERM (信號終止)發送到程序的終止請求。
每個庫實現可以提供可以與此函數一起使用的附加 信號值 宏常量。
請注意,並非所有運行環境都需要生成自動信號,即使在上述特定情況下也是如此,盡管所有運行環境都必須通過顯式調用生成的信號來生成 提高 功能。 - FUNC
-
指向函數的指針。這可以是程序員定義的函數,也可以是以下預定義函數之一:
SIG_DFL 默認處理:信號由該特定信號的默認操作處理。 SIG_IGN 忽略信號:忽略信號。
如果是一個函數,它應該遵循以下原型(使用C鏈接):
void handler_function (int parameter);
返回值
返回類型與參數func的類型相同。
如果請求成功,則該函數返回指向特定處理函數的指針,該函數在調用之前負責處理該信號(如果有的話)。或者SIG_DFL 要么 SIG_IGN如果在調用之前信號由默認處理程序處理或被忽略,則相應地。
如果該功能未能成功注冊新的信號處理程序,則返回SIG_ERR 和 錯誤號 可以設置為正值。
例
/* signal example */
#include <stdio.h> /* printf */
#include <signal.h> /* signal, raise, sig_atomic_t */
sig_atomic_t signaled = 0;
void my_handler (int param)
{
signaled = 1;
}
int main ()
{
void (*prev_handler)(int);
prev_handler = signal (SIGINT, my_handler);
/* ... */
raise(SIGINT);
/* ... */
printf ("signaled is %d.\n",signaled);
return 0;
}
#include <signal.h> typedef void (*sighandler_t)(int); sighandler_t signal(int signum, sighandler_t handler);
第一個參數是要捕捉的信號(查看信號:kill -l,9號SIGKILL信號不能被捕捉);
第二個參數表示我們要對信號進行的處理方式。
信號的處理方式一般有三種:
(1)忽略此信號(SIG_IGN):
#include<stdio.h>
#include<signal.h>
int main()
{
signal(2,SIG_IGN);
while(1)
{
printf("2333\n");
sleep(1);
}
return 0;
}
當執行程序時,陷入死循環,此時按下Ctrl+c進程並不會停止,因為我們對Ctrl+c產生的2號SIGINT信號采取了忽略處理,若要停止進程可用Ctrl+z(SIGQUIT);
運行結果如下:

(2)執行該信號的默認處理動作(SIG_DFL):
#include<stdio.h>
#include<signal.h>
int main()
{
signal(2,SIG_DFL);
while(1)
{
printf("2333\n");
sleep(1);
}
return 0;
}
當執行程序時,同樣是死循環,此時按下Ctrl+z進程停止,因為我們對2號信號采取默認動作處理,系統默認2號信號終止進程。
(3提供一個信號處理函數,要求內核在處理該信號時切換到用戶態執行這個處理函數,這種方式稱為捕捉(catch)一個信號:
#include<stdio.h>
#include<signal.h>
void handler(int signo)//自定義一個函數處理信號
{
printf("catch a signal:%d\n:",signo);
}
int main()
{
signal(2,handler);
while(1)
{
printf("1111\n");
sleep(1);
}
return 0;
}
運行結果如圖:

一些常用的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專用,進程超過資源限制的時候發 |
