Linux中的信号

Linux中的信号

简单讲讲Linux中的信号

信号的概念

信号(signal)是软件中断,是进程之间相互传递消息的一种方法,用于通知进程发生了事件,但是,不能给进程传递任何数据

信号产生的原因有许多种,在Linux中可以使用kill或者killall命令发送特定信号

信号的类型

Linux下常见的信号如下,其中加粗的需要重点关注

默认处理动作:

  • A 缺省的动作是终止进程。
  • B 缺省的动作是忽略此信号,将该信号丢弃,不做处理。
  • C 缺省的动作是终止进程并进行内核映像转储(core dump),内核映像转储是指将进程数据在内存的映像和进程在内核结构中的部分内容以一定格式转储到文件系统,并且进程退出执行,这样做的好处是为程序员 提供了方便,使得他们可以得到进程当时执行时的数据值,允许他们确定转储的原因,并且可以调试他们的程序。
  • D 缺省的动作是停止进程,进入停止状态的程序还能重新继续,一般是在调试的过程中。
  • E 信号不能被捕获。
  • F 信号不能被忽略。
信号名 信号值 默认处理动作 发出信号的原因
SIGHUP 1 A 终端挂起或者控制进程终止
SIGINT 2 A 键盘中断Ctrl+c
SIGQUIT 3 C 键盘的退出键被按下
SIGILL 4 C 非法指令
SIGABRT 6 C 由abort(3)发出的退出指令
SIGFPE 8 C 浮点异常
SIGKILL 9 AEF 采用kill -9 进程编号 强制杀死程序
SIGSEGV 11 C 无效的内存引用
SIGPIPE 13 A 管道破裂,写一个没有读端口的管道。
SIGALRM 14 A 由alarm()发出的信号
SIGTERM 15 A 采用“kill 进程编号”或“killall程序名”通知程序。
SIGUSR1 10 A 用户自定义信号1
SIGUSR2 12 A 用户自定义信号2
SIGCHLD 17 B 子进程结束信号
SIGCONT 18 进程继续(曾被停止的进程)
SIGSTOP 19 DEF 终止进程
SIGTSTP 20 D 控制终端(tty)上按下停止键
SIGTTIN 21 D 后台进程企图从控制终端读
SIGTTOU 22 D 后台进程企图从控制终端写

信号的处理

一般而言,进程对信号的处理方法有三种:

  1. 对该信号的处理采用系统的默认操作,大部分的信号的默认操作是终止进程

  2. 设置中断的处理函数,收到信号后,由该函数来处理

  3. 忽略某个信号,对该信号不做任何处理,就像未发生过一样

1
2
3
4
5
6
7
8
9
// signal函数可以用来设置程序对信号的处理方式
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
// signum表示信号的编号
// 参数handler表示对于信号的处理方式,有如下三种情况
// - SIG_DFL:signal.h 预制的宏定义,用于恢复信号的处理方式为默认处理方式
// - 一个自定义的信号处理函数 handler,执行函数内容
// - SIG_IGN:忽略参数signum所指代的信号

信号的作用

信号的作用顾名思义,给出一个一个信号,起到一个通知的作用表示发生了某件事情,这里要注意它只起到通知,不起到告知具体信息

比如说:

服务程序运行在后台,如果想让中止它,杀掉不是个好办法,因为程序被杀的时候,程序突然死亡,没有安排善后工作。

如果向服务程序发送一个信号,服务程序收到这个信号后,调用一个函数,在函数中编写善后的代码,程序就可以有计划的退出。

再比如说:

向服务程序发送0的信号,可以检测程序是否存活。

信号应用示例

屏蔽信号

一般在需要编辑一个服务器程序的时候,会先将所有的信号进行屏蔽,防止运行期间程序被一些预期之外的信号干扰

1
2
3
for (int i = 1; i < 65; ++i) {
		signal(i, SIG_IGN);
}

针对特定信号自行设定处理方式

比如针对ctrl + C以及kill + 进程号或者killall + 进程名这种目的是中断进程的信号。

其中ctrl + C发送的信号是SIGINT信号,编号为2;kill + 进程号或者killall + 进程名发送的信号是SIGTERM信号,编号为15

⚠️:另外还有一种方式会中断程序,即通过kill -9 进程号的方式发送编号为9的信号,此信号不能被捕获也不能被忽略,当发送这个信号时程序会直接停止运行,是强制关闭进程的一种措施

设置SIGINT和SIGTERM两个信号的处理函数,这两个信号可以使用同一个处理函数,函数的代码是在程序运行结束前释放资源,增强程序的健壮性,完整程序如下:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
#include <iostream>
#include <signal.h>

void EXIT(int sig) {
    printf("收到了信号%d, 即将退出\n", sig);
    /** 这里应该填充程序的资源释放代码
     * 
     * 
     */
    exit(0);    // 进程退出
}

int main(int argc, char** argv) {
    for (int i = 1; i < 65; ++i) {
        signal(i, SIG_IGN);
    }
    signal(SIGINT, EXIT);
    signal(SIGTERM, EXIT);

    while (true) {
        printf("模拟执行一次任务,每次任务两秒钟\n");
        sleep(2);
    }

    return 0;
}

发送信号

Linux操作系统提供了killkillall命令向程序发送信号,C语言也提供了相对应的库函数,用于在程序中向其它进程或者线程发送信号。

1
2
3
4
#include <sys/types.h>
#include <signal.h>

int kill(pid_t pid, int sig);

这里sig参数与之前终端发送的sig编号相同,要注意一下参数pid一共会有三种情况:

  • pid>0将信号传给进程号为pid的进程
  • pid=0将信号传给和目前进程相同进程组的所有进程,常用于父进程给子进程发送信号,注意,发送信号者进程也会收到自己发出的信号
  • pid=-1将信号广播传送给系统内所有的进程,例如系统关机时,会向所有的登录窗口广播关机信息

返回值说明:

  • 成功执行时,返回0

  • 失败返回-1,errno被设为以下的某个值:

    • EINVAL:指定的信号码无效(参数 sig 不合法)

    • EPERM:权限不够无法传送信号给指定进程

    • ESRCH:参数 pid 所指定的进程或进程组不存在

0%