信号是由操作系统传给进程的中断,会提早终止一个程序。在 UNIX、LINUX、Mac OS X 或 Windows 系统上,可以通过按 Ctrl+C 产生中断。 有些信号不能被程序捕获,但是下表所列信号可以在程序中捕获,并可以基于信号采取适当的动作。这些信号是定义在 C++ 头文件

信号 描述
SIGABRT 程序的异常终止,如调用 abort
SIGFPE 错误的算术运算,比如除以零或导致溢出的操作。
SIGILL 检测非法指令。
SIGINT 程序终止(interrupt)信号。
SIGSEGV 非法访问内存。
SIGTERM 发送到程序的终止请求。

signal() 函数

C++ 信号处理库提供了signal函数,用来捕获突发事件。以下是 signal() 函数的语法:

  1. void (*signal (int sig, void (*func)(int)))(int);
  2. void(*signal(int, void (*)(int)))(int);

这个看起来有点费劲,以下语法格式更容易理解:

  1. signal(registered signal, signal handler)

这个函数接收两个参数:

  • 第一个参数是一个整数,代表了信号的编号;
  • 第二个参数是一个指向信号处理函数的指针。

让我们编写一个简单的 C++ 程序,使用 signal() 函数捕获 SIGINT 信号。不管您想在程序中捕获什么信号,您都必须使用signal函数来注册信号,并将其与信号处理程序相关联。看看下面的实例:

  1. #include <iostream>
  2. #include <csignal>
  3. #include <unistd.h>
  4. using namespace std;
  5. void signalHandler(int signum)
  6. {
  7. cout << "Interrupt signal (" << signum << ") received.\n";
  8. // 清理并关闭 终止程序
  9. exit(signum);
  10. }
  11. int main()
  12. {
  13. // 注册信号 SIGINT 和信号处理程序
  14. signal(SIGINT, signalHandler);
  15. while (1)
  16. {
  17. cout << "Going to sleep...." << endl;
  18. sleep(1);
  19. }
  20. return 0;
  21. }
  22. /*
  23. Going to sleep....
  24. Going to sleep....
  25. Going to sleep....
  26. Going to sleep....
  27. Going to sleep....
  28. Going to sleep....
  29. Going to sleep....
  30. Going to sleep....
  31. Going to sleep....
  32. Going to sleep....
  33. Going to sleep....
  34. */

raise() 函数

您可以使用函数raise()生成信号,该函数带有一个整数信号编号作为参数,语法如下:

  1. int raise (signal sig);

在这里,sig是要发送的信号的编号,这些信号包括:SIGINT、SIGABRT、SIGFPE、SIGILL、SIGSEGV、SIGTERM、SIGHUP。以下是我们使用 raise() 函数内部生成信号的实例:

  1. #include <iostream>
  2. #include <csignal>
  3. //Linux 用 #include <unistd.h> 和 sleep(),Windos 用 #include <windows.h> 和 Sleep()。
  4. // Sleep 是区分大小写的,有的编译器是大写,有的是小写。
  5. // Sleep 括号里的时间,在 Windows 下是以毫秒为单位,而 Linux 是以秒为单位
  6. #include <unistd.h>
  7. using namespace std;
  8. void signalHandler(int signum)
  9. {
  10. cout << "Interrupt signal (" << signum << ") received.\n";
  11. // 清理并关闭 终止程序
  12. exit(signum);
  13. }
  14. int main()
  15. {
  16. int i = 0;
  17. // 注册信号 SIGINT 和信号处理程序
  18. signal(SIGINT, signalHandler);
  19. while (++i)
  20. {
  21. cout << "Going to sleep...." << endl;
  22. if (i == 3)
  23. {
  24. raise(SIGINT);
  25. }
  26. sleep(1);
  27. }
  28. return 0;
  29. }
  30. /*
  31. Going to sleep....
  32. Going to sleep....
  33. Going to sleep....
  34. Interrupt signal (2) received.
  35. */