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.