Unix/Linux的信号处理
信号(signal)就是Unix/Linux系统中最常见的一种软件中断的方式。中断 就是 程序中止当前正在执行的代码,转为执行其他代码的过程。中断 分为软件中断和硬件中断。软件中断的主要方式就是 信号。
(内存用内存地址管,进程用pid,文件用i节点或是文件描述符。都是用整数管理)
信号本质就是一个非负的整数,不同的值可以代表不同的情况。信号 都有一个宏名称,以SIG开头,比如:信号2 就是ctrl+c 的名字就叫SIGINT,宏名称的定义在POSIX规范中,而值不保证一致。因此,编程时,信号都是使用宏名称而不是值,否则可能出现不一致。
信号在Unix和Linux中是不同的,Unix是1-48,Linux 1-64 ,但中间不保证连续。信号0有特殊用途,没有实际的意义。
信号分为可靠信号和不可靠信号,不可靠信号在Linux系统中是 1-31,特点是不支持排队,因此多个相同信号同时到来时,可能出现信号的丢失。可靠信号是34-64,特点是 支持排队,因此不会丢失。
命令 kill -l 可以查看系统都有哪些信号。每种信号都有来源和处理方式。
信号是无法确定何时到来,在程序中无法判断信号的到来时间。
信号的处理方式:
1.默认处理方式 每个信号都有默认处理方式,如果不改变信号的处理方式,就采用默认处理方式。默认处理大多数就是退出进程。
2.忽略信号,不做任何的处理,就像信号没有来过一样。
3.自定义处理,程序员需要写一个信号处理函数,然后让信号的处理方式改为信号处理函数。(重点)
注:有些信号无法自定义或忽略,比如:信号9
信号可以在进程之间互发。信号的发送有权限限制,当前用户只能给自己的进程发信号,root用户可以给所有进程发信号。(安全)
如何设置信号的处理方式:
函数signal()或函数sigaction()可以
sigaction()就是增强版的signal(),但在应用中,signal()足够了,因此sigaction()介绍一下.
信号的重点:
void(*func)(int) signal(int signum, void(*func)(int))
功能:设置某个信号的处理方式为默认/忽略/自定义函数
参数:signum第一个参数:设置哪个信号的处理方式
第二个参数就是函数指针,可以是SIG_DFL,SIG_IGN或是一个自定义的函数
返回值:成功返回之前的处理方式(一般用不上),失败返回 SIG_ERR
信号处理的基本步骤:1.#include一个头文件
2.写一个信号的处理函数,格式:void func(int signo)//函数名可变,参数是信号的值
3.在主函数调用signal()改变指定信号的处理方式
父子进程之间的信号处理方式
如果是fork()创建的子进程,子进程将完全复制父进程的信号处理方式。父子进程的信号处理方式一样。
如果是vfork()+execl()创建的子进程,父进程默认,子进程默认;父进程忽略,子进程也忽略;父进程采用自定义处理函数,子进程改为默认处理。