C++ 信號處理
信號是由操作系統傳給進程的中斷,會提早終止一個程序。 在UNIX、LINUX、Mac OS X 或Windows 系統上,可以通過按Ctrl+C 產生中斷。
有些信號不能被程序捕獲,但是下表所列信號可以在程序中捕獲,並可以基於信號採取適當的動作。 這些信號是定義在C++ 頭文件<csignal> 中。
信号 | 描述 |
---|---|
SIGABRT | 程序的异常终止,如调用abort。 |
SIGFPE | 错误的算术运算,比如除以零或导致溢出的操作。 |
SIGILL | 检测非法指令。 |
SIGINT | 接收到交互注意信号。 |
SIGSEGV | 非法访问内存。 |
SIGTERM | 发送到程序的终止请求。 |
signal() 函數
C++信號處理庫提供了signal函數,用來捕獲突發事件。 以下是signal() 函數的語法:
void (*signal (int sig, void (*func)(int)))(int);
這個函數接收兩個參數:第一個參數是一個整數,代表了信號的編號;第二個參數是一個指向信號處理函數的指針。
讓我們編寫一個簡單的C++ 程序,使用signal() 函數捕獲SIGINT 信號。 不管您想在程序中捕獲什麼信號,您都必須使用signal函數來註冊信號,並將其與信號處理程序相關聯。 看看下面的實例:
#include <iostream> #include <csignal> using namespace std; void signalHandler( int signum ) { cout << "Interrupt signal (" << signum << ") received.\n"; // 清理并关闭 // 终止程序 exit(signum); } int main () { // 注册信号 SIGINT 和信号处理程序 signal(SIGINT, signalHandler); while(1){ cout << "Going to sleep...." << endl; sleep(1); } return 0; }
當上面的代碼被編譯和執行時,它會產生下列結果:
Going to sleep.... Going to sleep.... Going to sleep....
現在,按Ctrl+C 來中斷程序,您會看到程序捕獲信號,程序打印如下內容並退出:
Going to sleep.... Going to sleep.... Going to sleep.... Interrupt signal (2) received.
raise() 函數
您可以使用函數raise()生成信號,該函數帶有一個整數信號編號作為參數,語法如下:
int raise (signal sig);
在這裡,sig是要發送的信號的編號,這些信號包括:SIGINT、SIGABRT、SIGFPE、SIGILL、SIGSEGV、SIGTERM、SIGHUP。以下是我們使用raise() 函數內部生成信號的實例:
#include <iostream> #include <csignal> using namespace std; void signalHandler( int signum ) { cout << "Interrupt signal (" << signum << ") received.\n"; // 清理并关闭 // 终止程序 exit(signum); } int main () { int i = 0; // 注册信号 SIGINT 和信号处理程序 signal(SIGINT, signalHandler); while(++i){ cout << "Going to sleep...." << endl; if( i == 3 ){ raise( SIGINT); } sleep(1); } return 0; }
當上面的代碼被編譯和執行時,它會產生下列結果,並會自動退出:
Going to sleep.... Going to sleep.... Going to sleep.... Interrupt signal (2) received.