你的位置:首页 > 软件开发 > 操作系统 > 进程间通信(五)—信号

进程间通信(五)—信号

发布时间:2016-07-10 00:00:05
我会用几篇博客总结一下在Linux中进程之间通信的几种方法,我会把这个开头的摘要部分在这个系列的每篇博客中都打出来进程之间通信的方式管道消息队列信号信号量共享存储区套接字(socket)进程间通信(三)—信号量传送门:http://www.cnblogs.co ...

进程间通信(五)—信号

我会用几篇博客总结一下在Linux中进程之间通信的几种方法,我会把这个开头的摘要部分在这个系列的每篇博客中都打出来

进程之间通信的方式

  • 管道
  • 消息队列
  • 信号
  • 信号量
  • 共享存储区
  • 套接字(socket)

进程间通信(三)—信号量传送门:http://www.cnblogs.com/lenomirei/p/5649792.html

进程间通信(二)—消息队列传送门:http://www.cnblogs.com/lenomirei/p/5642575.html

进程间通信(一)—管道传送门:http://www.cnblogs.com/lenomirei/p/5636339.html

我感觉这么写下去越来越不像进程间的通信了,更像是进程间的打招呼。。。信号就是这样的,某某(这个某某有很多可能性)给进程一个信号,进程就会在适当的情况下处理这个信号这说明进程可能不会立即处理信号),什么是适当的时候呢?比如说中断返回的时候,或者内核态返回用户态的时候(这个情况出现的比较多)等等(推荐本书《Linux内核设计与实现》,里面讲过)。。。这个也不是本篇的主题就不多述了。

首先来说信号是怎么产生的

  • 由硬件产生,别入从键盘敲入组合键发送一个信号,常用的Ctrl+C就可以给前台进程发送。
  • 由进程发送(或者说是由软件产生),比如我们可以在shell进程下输入kill命令给一个进程发送信号(命令:kill -信号标号 PID)
  • 异常,当异常的发生的时候肯定是会发送信号的

然后说信号的处理方式,谁来处理信号?肯定是操作系统来,难不成还是程序员么。文章一开头就说了,号不一定会被立即处理,操作系统不会为了处理一个信号而把当前正在运行的进程挂起(切换进程)或者杀掉(肯定不会杀掉啊,难道看见一个信号就杀害一个无辜群众么),挂起(进程切换)的话消耗太大了,如果不是紧急信号,可能是不会立即处理的。操作系统多选择在内核态切换回用户态的时候处理信号,这样就利用两者的切换来处理了(不用单独进行进程切换以免浪费时间)。

总归是不能避免的,因为很有可能在睡眠的进程就接收到信号,操作系统肯定不愿意切换当前正在happy地跑着的进程,于是就得把信号储存啊,因为是进程收到的信号,所以把信号储存在进程唯一的PCB(就是task_struct)当中。struct sigpending pending;字段就是存放信号的信号表,之后会解释pending。

 信号的处理过程

进程间通信(五)—信号

所有的信号可以通过kill -l 命令查看

进程间通信(五)—信号

需要注意的几点

  • 没有0号信号
  • 没有32,33号信号
  • 从31号信号分开,前面的是普通信号,后面的是实时信号,本篇介绍的是普通信号
  • 信号和前面的标号是一致的,可以使用标号,也可以使用宏名称

信号的处理方式有三种

  • 忽略(就是这么6,你发啥我都不理你)
  • 默认处理方式,操作系统设定的默认处理方式,会终止一个进程,就是说接到信号就把这个进程干掉了
  • 自定义处理方式,想干什么还是得我说了算,你需要一个signal函数
    • 函数原型:sighandler_t signal(int signum, sighandler_t handler);
    • 头文件:#include <signal.h>
    • 参数解析:
      • 第一个是信号标号,给数字就可以啦
      • 关键是第二个参数这里,sighandler_t是一个typedef来的,主要是为了可读性,原型是void (*)(int)函数指针啦,int的参数会被设置成signum
      • 我有用到这个函数,可以在我的测试用例中看一下用法

 

  • 相关函数解析

这次就没有创建函数什么的了,主要是信号相关的一些操作函数,但是由于比较多和繁杂,不好每一个都写测试用例看输出结果,最后的测试程序只用了一部分函数,并没有全部使用到

前面说了用kill命令可以给进程发信号,可是我想用C语言编写程序发,别怕!你需要下面的函数(raise函数只能给进程本身发信号

  • 函数原型:int raise(int signo);
  • 头文件:#include <signal.h>
  • 参数解析:
    • 就一个参数就是信号了,可以用标号,也可以用宏名

或者你说不想光给自己发信号,光自己是很没意思,那么看下面这个函数

  • 函数原型:int kill(pid_t pid, int signo);
  • 头文件:#include <signal.h>
  • 参数解析:
    • 第一个参数是pid进程号,你得让操作系统知道你要给哪个进程发信号,给自己发也是可以的哦
    • 第二个参数是信号。。。呃,准确的说是哪个信号,可以用标号,也可以用宏名

看这个信号(6) SIGABRT,这个信号可以让进程异常终止,他有一个对应的函数

  • 函数原型:void abort(void);
  • 头文件:#include <stdlib.h>
  • 参数解析:
    • 这根本就没有参数嘛!那就写点其他的。
    • 该函数没有返回值,因为该函数执行绝对不会失败(为什么?杀个进程而已,谈什么失败(一刀不行就两刀!!(玩笑))),和exit函数一样绝对不会失败(重要!说两遍!)和exit函数不同的地方是,exit会设置退出码。

看这个信号(14) SIGALRM,这个信号是闹钟信号,它可以由这个函数发送

  • 函数原型:unsigned int alarm(unsigned int seconds);
  • 头文件:#include <unistd.h>
  • 参数解析:
    • 参数都写的这么清楚了!等待多少秒之后就会发送SIGALRM信号给当前进程。

接下来说明一下阻塞信号,就是(pending)了

  • 阻塞信号(pending signal)

之前提到过,可以忽略一个信号,那么我说,阻塞和忽略是不一样的阻塞是进程收不到该信号,忽略是进收到该信号,却不做出任何反应

实际执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未决(Pending)。进程可以选择阻塞(Block )某个信号。被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作。

task_struct中有类似字段,之前说的信号可能不会立即被执行就会存储在pending表里面

进程间通信(五)—信号

其中一旦信号被block,当有信号产生的时候会一直pending,因为未决就是未处理的意思,block到不了就处理不了,pending会一直有该位

以为PCB中使用32位来表示上图中的block和pending表,所以非实时信号就算发送多个,也只显示一个。实时信号会排队,有几个来就有几个排队

block表:某位为0表示该位对应标号的信号未被阻塞,为1表示阻塞

pending表:某位为0表示信号还未产生或者已经被处理

上图中2号信号两个表同时为1表示产生了2号信号,但因为2号信号被阻塞所以一直未决。

介绍几个操作这两张表的函数

  • 函数原型:

    int sigemptyset(sigset_t *set);用于初始化一个信号集(新创建出来的sigset_t对象都要先执行这个函数再去操作)

    原标题:进程间通信(五)—信号

    关键词:进程

*特别声明:以上内容来自于网络收集,著作权属原作者所有,如有侵权,请联系我们: admin#shaoqun.com (#换成@)。