进程间通信特点(进程间通信五)
导航博客园首页发新随笔发新文章联系订阅管理 |
进程间通信(五)—信号 我会用几篇博客总结一下在Linux中进程之间通信的几种方法,我会把这个开头的摘要部分在这个系列的每篇博客中都打出来 进程之间通信的方式
进程间通信(三)—信号量传送门: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内核设计与实现》,里面讲过)。。。这个也不是本篇的主题就不多述了。 首先来说信号是怎么产生的!
然后说信号的处理方式,谁来处理信号?肯定是操作系统来,难不成还是程序员么。文章一开头就说了,信号不一定会被立即处理,操作系统不会为了处理一个信号而把当前正在运行的进程挂起(切换进程)或者杀掉(肯定不会杀掉啊,难道看见一个信号就杀害一个无辜群众么),挂起(进程切换)的话消耗太大了,如果不是紧急信号,可能是不会立即处理的。操作系统多选择在内核态切换回用户态的时候处理信号,这样就利用两者的切换来处理了(不用单独进行进程切换以免浪费时间)。 总归是不能避免的,因为很有可能在睡眠的进程就接收到信号,操作系统肯定不愿意切换当前正在happy地跑着的进程,于是就得把信号储存啊,因为是进程收到的信号,所以把信号储存在进程唯一的PCB(就是task_struct)当中。struct sigpending pending;字段就是存放信号的信号表,之后会解释pending。 信号的处理过程
所有的信号可以通过kill -l命令查看
需要注意的几点
信号的处理方式有三种
这次就没有创建函数什么的了,主要是信号相关的一些操作函数,但是由于比较多和繁杂,不好每一个都写测试用例看输出结果,最后的测试程序只用了一部分函数,并没有全部使用到 前面说了用kill命令可以给进程发信号,可是我想用C语言编写程序发,别怕!你需要下面的函数(raise函数只能给进程本身发信号)
或者你说不想光给自己发信号,光自己是很没意思,那么看下面这个函数
看这个信号(6) SIGABRT,这个信号可以让进程异常终止,他有一个对应的函数
看这个信号(14) SIGALRM,这个信号是闹钟信号,它可以由这个函数发送
接下来说明一下阻塞信号,就是(pending)了 之前提到过,可以忽略一个信号,那么我说,阻塞和忽略是不一样的,阻塞是进程收不到该信号,忽略是进收到该信号,却不做出任何反应 实际执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未决(Pending)。进程可以选择阻塞(Block )某个信号。被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作。 task_struct中有类似字段,之前说的信号可能不会立即被执行就会存储在pending表里面
其中一旦信号被block,当有信号产生的时候会一直pending,因为未决就是未处理的意思,block到不了就处理不了,pending会一直有该位 以为PCB中使用32位来表示上图中的block和pending表,所以非实时信号就算发送多个,也只显示一个。实时信号会排队,有几个来就有几个排队 block表:某位为0表示该位对应标号的信号未被阻塞,为1表示阻塞 pending表:某位为0表示信号还未产生或者已经被处理 上图中2号信号两个表同时为1表示产生了2号信号,但因为2号信号被阻塞所以一直未决。 介绍几个操作这两张表的函数
如何阻塞一个信号?用下面的函数
如何获取当前pending表中的信息?你需要它~
事已至此,基本操作就说完了,废话少说,show me the code 功能看结果图:先贴结果图,一开始没有任何信号,所以pending表中全是0,我通过Ctrl C传入2号信号,看到pending表中有2号被置位了,经过10秒取消阻塞,2号信号被处理(经过我自定义的函数)
我的程序只有一个文件server.c
1 #include <stdio.h> 2 #include <sys/signal.h> 3 #include <sys/types.h> 4 #include <signal.h> 5 6 7 8 void func(int num) 9 { 10 printf("catch signal number is %d",num); 11 12 } 13 14 15 void printfpendingsignal(sigset_t *set) 16 { 17 int i; 18 for(i=1;i<32; i) 19 { 20 if(sigismember(set,i)) 21 { 22 printf("1"); 23 24 } 25 else 26 { 27 printf("0"); 28 } 29 } 30 printf("\n"); 31 } 32 33 34 int main 35 { 36 sigset_t s,p,o; 37 signal(SIGINT,func); 38 sigemptyset(&s); 39 sigemptyset(&p); 40 sigemptyset(&o); 41 sigaddset(&s,SIGINT); 42 sigprocmask(SIG_SETMASK,&s,&o); 43 int count=0; 44 while(1) 45 { 46 sigpending(&p); 47 printfpendingsignal(&p); 48 sleep(1); 49 if(count ==10) 50 { 51 printf("recover!\n"); 52 sigprocmask(SIG_SETMASK,&o,NULL); 53 } 54 } 55 return 0; 56 } posted on 2016-07-09 23:16 leno米雷阅读(...) 评论(...)编辑收藏 刷新评论刷新页面返回顶部 |
免责声明:本文仅代表文章作者的个人观点,与本站无关。其原创性、真实性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容文字的真实性、完整性和原创性本站不作任何保证或承诺,请读者仅作参考,并自行核实相关内容。文章投诉邮箱:anhduc.ph@yahoo.com