C++ 信号处理
信号是操作系统传递给进程的中断,可以提前终止程序。在 UNIX、LINUX、Mac OS X 或 Windows 系统上,您可以通过按 Ctrl+C 来生成中断。
有些信号无法被程序捕获,但以下列出的信号您可以在程序中捕获,并根据信号采取适当的操作。这些信号定义在 C++ 头文件 <csignal> 中。
Sr.No | 信号 &描述 |
---|---|
1 | SIGABRT 程序异常终止,例如调用abort。 |
2 | SIGFPE 错误的算术运算,例如除以零或导致溢出的运算。 |
3 | SIGILL 检测到非法指令。 |
4 | SIGINT 收到交互式注意信号。 |
5 | SIGSEGV 无效的存储访问。 |
6 | 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."; // 清理并关闭此处的内容 // 终止程序 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."; // 清理并关闭此处的内容 // 终止程序 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.