信号的概念
为了理解信号,先从我们最熟悉的场景说起:
1. ⽤户输⼊命令,在Shell下启动⼀个前台进程。
2. ⽤户按下Ctrl-C,这个键盘输⼊产⽣⼀个硬件中断。
3. 如果CPU当前正在执⾏这个进程的代码,则该进程的⽤户空间代码暂停执⾏,CPU从⽤户态 切换到内核态处理硬件中断。
4. 终端驱动程序将Ctrl-C解释成⼀个SIGINT信号,记在该进程的PCB中(也可以说发送了⼀ 个SIGINT信号给该进程)。
5. 当某个时刻要从内核返回到该进程的⽤户空间代码继续执⾏之前,⾸先处理PCB中记录的信号,发现有⼀个SIGINT信号待处理,⽽这个信号的默认处理动作是终⽌进程,所以直接终⽌进程⽽不再返回它的⽤户空间代码执⾏。
需要注意的是:键盘上的组合键形成的信号只能用于前台进程
目前来说,前台进程随时随地都可以收到一个信号,因为你在这个进程运行的任何时刻,你都可以出入“Ctrl-C”终止该进程。也就是说,信号对于进程来说是异步的。
对于信号的种类的和多少,可以再Linux中断输入“kill -l”显示。
每个信号都是一个宏定义,具体的宏值为多少请百度。
信号的产生
产生信号的方法有四种:
1. 对于前台进程来说,通过键盘的组合键就可以;
2. 计算机的软/硬件中断或异常。
硬件异常产⽣信号,这些条件由硬件检测到并通知内核,然后内核向当前进程发送适当的信号。例如当前进程执⾏了除以0的指令,CPU的运算单元会产⽣异常,内核将这个异常解释 为SIGFPE信号发送给进程。再⽐如当前进程访问了⾮法内存地址,,MMU会产⽣异常,内核 将这个异常解释为SIGSEGV信号发送给进程。
#include <stdio.h>
#include <signal.h>
#include <sys/types.h>
#include <stdlib.h>
int main()
{
int i=0;
int ret=10/i;
return 0;
}
[dlm@localhost temp]$
3. signal函数,但严格来说signal并不是一个产生信号的函数,而是一个改变信号默认动作的函数,signal函数可以对于指定的信号更改他的默认动作来达到自定义信号动作的效果。
#include <stdio.h>
#include <signal.h>
#include <sys/types.h>
#include <stdlib.h>
#include "mysleep.h"
void Handler(