Linux的信号(signal)

0

本文涉及:

  • Linux信号
  • Linux信号类型
  • 信号处理

信号的基本概念

信号(signal)是软件中断,是进程之间通信的一种方式,用以通知目标进程某个状态的改变或系统异常,但不能给进程传递任何数据。它是在软件层面对中断机制的模拟,是一种异步通信的方式。

信号的类型

信号名 信号值 默认处理动作 发出信号的原因
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(2)发出的信号
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 后台进程企图从控制终端写

信号的几种默认处理动作:

  • A 缺省的动作是终止进程
  • B 缺省的动作是忽略此信号,将该信号丢失,不做处理
  • C 缺省的动作是终止进程并进行内核映像转储(core dump),内核映像转储是指将进程数据在内存的映像和进程在内核结构中的部分内容以一定格式转储到文件系统,并且进程退出执行,这样做的好处是为程序员提供了方便,使得他们可以得到进程当时执行时的数据值,允许他们确定转储的原因,并且可以调试他们的程序。
  • D 缺省的动作是停止进程,进入停止状态的进程还能继续运行,一般在调试阶段。
  • E 信号不能捕获
  • F 信号不能被忽略

信号的处理

进程对信号的处理方法有3种:

  • 对信号的处理采用系统默认操作,大部分的信号的默认操作时终止操作
  • 设置中断的处理函数,收到信号后,由该函数处理
  • 忽略某个信号,对该信号不作任何处理

signal函数可以设置进程对信号的处理方式。

函数声明:

1
sighandler_t signal(int signum,sighandler_t handler);

参数signum表示信号的编号。

参数handler表示信号的处理方式,有三种方式:

  • SIG_DFL 恢复参数signum所指信号的处理方式为默认。
  • 一个自定义的处理信号的函数,信号的编号为这个函数的参数。
  • SIG_IGN 忽略参数signum所指的信号。

信号的作用

服务程序运行在后台,如果想中止它,直接杀掉不是好办法,以为程序被杀死时,没有安排善后的工作。

如果向服务程序发送一个信号,服务程序接受到这个信号后,调用响应的处理函数,完成善后工作,程序就可以完美的退出了。

向进程发送0的信号,可以检测进程是否活着。

应用示例

在实际开发中,在main函数开始的位置,会先屏蔽掉所有的信号。

1
for(int i=1;i<=64;++i) signal(ii,SIG_IGN);

这么做的目的是不希望程序被打扰,然后设置需要用到的信号处理函数。

程序在运行的进程中,如果按Ctrl+c,将向程序发出SIGINT信号,编号是2。采用“kill 进程编号”或“killall 程序名”向程序发出的是SIGTERM信号,编号是15。采用“kill -9 进程编号”向程序发出的是SIGKILL信号,编号是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
27
28
29
#include <iostream>
#include <cstdlib>
#include <csignal>
#include <unistd.h>

void EXIT(int signum);//信号处理程序

int main()
{
for(int ii=1;ii<=64;++ii) signal(ii,SIG_IGN);//屏蔽所有信号

signal(SIGINT,EXIT); signal(SIGTERM,EXIT); // 设置SIGINT和SIGTERM的处理函数

while(1){//间隔1秒执行一次
std::cout << "test" <<std::endl;
sleep(1);
}

return 0;
}

void EXIT(int signum){
std::cout << "收到了" << signum << "信号,程序退出" << std::endl;
/*
*在此写入处理代码
*释放资源
*/
exit(0);
}

在一个编译并运行这段代码。

1
2
3
4
g++ -g test.cpp -o test
#编译
./test
#运行

运行效果

1

用Ctrl+c或者kill命令终止test进程,都可以体面的结束运行。

2

3

发送信号

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

函数声明:

1
int kill(pid_t pid,int sig);

kill函数将参数sig指定的信号给参数pid指定的进程。

参数pid的几种情况;

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

sig:准备发送的信号代码,假如其值为零则没有任何信号送出,但是系统会执行错误检查,通常会利用sig值为零来检验某个进程是否仍在运行。

返回值说明: 成功执行时,返回0;失败返回-1,error被设为以下的某个值。

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

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

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

-------------本文结束感谢您的阅读-------------