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.