一、信号
1、基本概念
(1)这里讲的信号与信号量不同,信号是进程给操作系统或进程的某种信息,让操作系统或者其他进程做出某种反应。
(2)信号是进程间通信机制中唯一的异步通信机制,一个进程不必通过任何操作来等待信号的到达,事实上,进程也不知道信号到底什么时候到达。进程之间可以互相通过系统调用kill发送软中断信号。内核也可以因为内部事件而给进程发送信号,通知进程发生了某个事件。
(3)信号机制除了基本通知功能外,还可以传递附加信息。
2、Linux下的信号列表
我们可以用kill -l命令查看Linux操作系统下的信号列表:
从上图可知,每个信号都有一个编号和一个宏定义名称,这些宏定义可以在signal.h中找到。
在这62个信号中,1–32号是普通信号,34–64号信号是实时信号,为了更加详细的了解各个信号,我们可以用 man 7 signal指令查看:
3、信号的产生条件
(1) 用户在终端按下某些键时,终端驱动程序会发送信号给前台进程,例如Ctrl-C产生SIGINT信号,Ctrl-\产生SIGQUIT信号,Ctrl-Z产生SIGTSTP信号(可使前台进程停止)
SIGINT的默认处理动作是终止进程,SIGQUIT的默认处理动作是终止进程并且Core Dump,所以我们可以写一个死循环,让后通过对这个进程发送SIGINT(ctrl-C)SIGQUIT(ctrl-\)信号来终止这进程。
我们写一个简单的死循环来验证一下:
1 #include<stdio.h>
2
3 int main()
4 {
5 while(1)
6 {
7 printf("hello word!\n");
8 sleep(1);
9 }
10 return 0;
11 }
运行结果:
这个程序每隔一秒打印出一条“hello word!”语句,因为是死循环,所以我们可以向前台发送Ctrl-C结束该进程。
(2)硬件异常产生信号,这些条件由硬件检测到并通知内核,然后内核向当前进程发送适当的信号。例如当前进程执行了除以0的命令,CPU的运算单元会产生异常,内核将这个异常解释 为SIGFPE信号发送给进程。
(3) 一个进程调用kill(2)函数可以发送信号给另一个进程。可以用kill(1)命令发送信号给某个进程,kill(1)命令也是调用kill(2)函数实现的,如果不明确指定信号则发送SIGTERM信号,该信号的默认处理动作是终止进程。
(4)软件条件产生。
4、信号的常见处理方式
(1)忽略此信号
(2)执行该信号的默认处理动作
(3)提供一个信号处理函数,要求内核在处理该信号时切换到用户态执行这个处理函数,这种方式称为捕捉一个信号。