C++ 信号处理
信号是操作系统传递给进程的中断,它可以提前终止程序。您可以通过在 UNIX、LINUX、Mac OS X 或 Windows 系统上按 Ctrl+C 来生成中断。
有些信号无法被程序捕获,但您可以在程序中捕获以下信号列表,并可以根据这些信号采取适当的操作。这些信号在 C++ 头文件 <csignal> 中定义。
| 先生编号 | 信号及描述 |
|---|---|
| 1 | SIGABRT 程序异常终止,例如调用abort。 |
| 2 | SIGFPE 错误的算术运算,例如除以零或导致溢出的运算。 |
| 3 | 西吉尔 检测到非法指令。 |
| 4 | 信号情报 收到交互式注意信号。 |
| 5 | 信号发生器 对存储的访问无效。 |
| 6 | 信号术语 发送到程序的终止请求。 |
signal() 函数
C++ 信号处理库提供了信号函数来捕获意外事件。以下是 signal() 函数的语法 -
void (*signal (int sig, void (*func)(int)))(int);
为简单起见,该函数接收两个参数:第一个参数为整数,表示信号编号,第二个参数为指向信号处理函数的指针。
让我们编写一个简单的 C++ 程序,在其中使用 signal() 函数捕获 SIGINT 信号。无论您想在程序中捕获什么信号,都必须使用信号函数注册该信号并将其与信号处理程序关联。检查以下示例 -
#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.
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";
// 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.