一.信号的定义
信号通信,其实就是内核向用户空间进程发送信号,只有内核才能发信号,用户空间进程不能发送信号。即信号间是不可以通过进程进行通信的,要先进入内核
内核可以发送多少种信号呢?
kill ‐l
命令:kill ‐9 pid
信号通信的框架
信号的发送(发送信号进程):kill、raise、alarm
信号的接收(接收信号进程) : pause()、 sleep、 while(1)
信号的处理(接收信号进程) :signal
二.信号的发送(发送信号进程)
kill:
所需头文件:
#include<signal.h>
#include<sys/types.h>
函数原型:int kill(pid_t pid, int sig);
参数:
函数传入值:pid
正数:要接收信号的进程的进程号
0:信号被发送到所有和pid进程在同一个进程组的进程
‐1:信号发给所有的进程表中的进程(除了进程号最大的进程外)
sig:信号
函数返回值:成功 0 出错 ‐1
kill函数的封装
a.out里面是一个while死循环,用封装好的kill去杀死进程
raise: 发信号给自己 == kill(getpid(), sig)
所需头文件:
#include<signal.h>
#include<sys/types.h>
函数原型:
int raise(int sig);
参数:
函数传入值:sig:信号
函数返回值:
成功 0 出错 ‐1
为什么没有函数打印出内容?
printf()是一个库函数,没有将缓存写到内核里面,需要加回车换行符才行
信号raise父子进程通信
if(waitpid(pid,NULL,0)==0)
waitpid(pid,NULL,0),0代表非阻塞状态,返回值为0表示子进程正在运行,如果子进程正在运行及就发送信号杀死子进程
8秒钟之前父进程在睡眠状态,子进程在停止状态
8秒后父进程运行杀死了子进程,子进程变成了僵尸进程,因为父进程一直在while循环没有空回收子进程
现在让父进程在while之前阻塞的回收子进程,这时8秒之后子进程被回收了,就不存在了,只有父进程在运行
alarm : 发送闹钟信号的函数
alarm 与 raise 函数的比较:
相同点:让内核发送信号给当前进程
不同点:
alarm 只会发送SIGALARM信号
alarm 会让内核定时一段时间之后发送信号, raise会让内核立刻发信号
所需头文件#include <unistd.h>
函数原型 unsigned int alarm(unsigned int seconds)
参数:
seconds:指定秒数
返回值:
成功:如果调用此alarm()前,进程中已经设置了闹钟时间,则 返回上一个闹钟时间的剩余时间,否则返回0。
出错:‐1
while循环没隔一秒打印,正常是答应到19才结束,现在用alarm函数在第七秒的时候就结束进程
三.信号的接收
接收信号的进程,要有什么条件?
要想使接收的进程能收到信号,这个进程不能结束 :
sleep
pause:进程状态为S(睡眠)
函数原型 int pause(void);无参数
函数返回值 成功:0,出错:‐1
使用pause函数之后去执行,可以发现我们的./a.out是正在睡眠的
可以通过ctrl+c结束进程,也可以ctrl+z 暂停进程
四.信号处理
收到信号的进程,应该怎样处理?
处理的方式:
1.进程的默认处理方式(内核为用户进程设置的默认处理方式)
A:忽略B:终止进程C: 暂停
2.自己的处理方式:
自己处理信号的方法告诉内核,这样你的进程收到了这个信号就会采用你自己的的处理方式、
所需头文件 #include <signal.h>
函数原型 void (*signal(int signum, void (*handler)(int)))(int);
函数传入值
signum:指定信号
handler
SIG_IGN:忽略该信号。
SIG_DFL:采用系统默认方式处理信号
自定义的信号处理函数指针
函数返回值
成功:设置之前的信号处理方式
出错:‐1
signal 函数有二个参数
第一个参数:一个整形变量(信号值)
第二个参数:一个函数指针,是我们自己写的处理函数;这个函数的返回值是一个函数指针。
SIGALRM(sigalrm) 的信号值是14
signal(14,myfun),第二个参数自定义的信号处理函数指针,它就会去执行函数里内容后再结束进程。如果用signal(14,SIG_IGN)就会忽略14这个信号继续执行程序,直到程序执行完成再退出,signal(14,SIG_DFL)就是执行14这个信号,也就是直接终止进程。
系统调用alarm函数进入内核,然后延时7秒,先去执行main函数里while的内容,7秒后因为signal用了第二个参数,所以会去执行myfun函数内容,执行完之后回到main主函数执行剩下的内容,执行完之后才结束。