Linux进程间通信——信号

一.信号的定义

信号通信,其实就是内核向用户空间进程发送信号,只有内核才能发信号,用户空间进程不能发送信号。即信号间是不可以通过进程进行通信的,要先进入内核

内核可以发送多少种信号呢?
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主函数执行剩下的内容,执行完之后才结束。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值