Cplusplus 简明教程

C++ Signal Handling

信号是由操作系统发送到进程中的中断,它可以使程序提前终止。你可以在 UNIX、LINUX、Mac OS X 或 Windows 系统上按下 Ctrl+C 来生成中断。

程序无法捕获的信号存在,但以下列表中列出的信号可以在程序中捕获,并可以根据信号采取适当的操作。这些信号在 C++ 头文件 <csignal> 中定义。

Sr.No

Signal & Description

1

SIGABRT 程序的异常终止,例如调用 abort

2

SIGFPE 错误的算术运算,例如除以零或导致溢出的运算。

3

SIGILL 检测到非法指令。

4

SIGINT 收到交互式关注信号。

5

SIGSEGV 对存储的无效访问。

6

SIGTERM 向程序发送的终止请求。

The signal() Function

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";

   // cleanup and close up stuff here
   // terminate program

   exit(signum);
}

int main () {
   // register signal SIGINT and signal handler
   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.

The raise() Function

您可以通过函数 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";

   // cleanup and close up stuff here
   // terminate program

   exit(signum);
}

int main () {
   int i = 0;
   // register signal SIGINT and signal handler
   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.