Linux 信号详解五(信号阻塞,信号未决)
信号在内核中的表示
执行信号的处理动作成为信号递达(Delivery),信号从产生到递达之间的状态称为信号未决(Pending)。进程可以选择阻塞(Block)某个信号。
被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作
注意:阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作
信号不会丢失,如果信号被阻塞,只会保持信号未决,但是信号不丢失
只要取消阻塞 信号依然会到来
①PCB进程控制块中有信号屏蔽状态字(block),信号未决状态字(pending)还有是否忽略标识
②信号屏蔽状态字(block):1代表阻塞,0代表不阻塞;信号未决状态字(pending):1代表未决,0代表信号递达
③向进程发送SIGINT,内核首先判断信号屏蔽状态字是否阻塞,如果信号屏蔽状态字阻塞,信号未决状态字(pengding)相应位置1;
若阻塞解除,信号未决状态字(pending)相应位置0,表示信号可以递达了。
④block状态字,pending状态都是64bit,分别代表Linux系统中的64个信号。例如SIGINT是2号信号,对应block状态字中的第二位
⑤block状态字用户可以读写,pending状态字用户只能读,这是新号的设计机制。
信号集操作函数(block状态字表示)
#include <signal.h>
int sigemptyset(sigset_t *set);
sigset_t *set实质上是8个字节大小的变量(8个字节64bit,代表Linux的64个信号)
sigemptyset将状态字置零
int sigfillset(sigset_t *set);
sigfillset将状态字全部置1(32号信号和33号信号除外)
int sigaddset(sigset_t *set,int signo);
将信号signo加入到信号集set中
int sigdelset(sigset_t *set,int signo);
将信号signo从信号集中删除
int sigismember(const sigset *set,int signo);
判断signo是否在信号集中
sigprocmask读取或者更改进程的信号屏蔽状态字(block)
#include <signal.h>
int sigprocmask(int how,const sigset_t *set,sigset * oset);
成功返回0,出错返回-
如果oset是非空指针,则读取进程的当前信号屏蔽状态字通过oset参数传出,如果set是非空指针,则更改进程的信号屏蔽状态字,参数how只是如何更改。
如果oset和set都是非空指针,则先将原来的信号屏蔽字备份到oset里,然后根据set和how参数更改信号屏蔽字。
how含义
--SIG_BLOCK set包含了我们希望添加到当前信号屏蔽字的信号,相当于mask=mask|set(位或运算)
--SIG_UNBLOCK set包含了我们希望从当前信号屏蔽字中解除阻塞的信号,相当于mask=mask^set(位异或运算)
--SIG_SETMASK 设置当前信号屏蔽字为set所指向的值,相当于mask=set
int sigpending(sigset_t *set);
获取信号未决状态字(pending)信息
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h> void printsigset(const sigset_t *pset)
{
int i = ;
//遍历64个信号,
for (; i < ; i++)
{
//信号从1开始 判断哪些信号在信号未决状态字中
if (sigismember(pset, i + ))
putchar('');
else
putchar('');
//fflush(stdout);
}
printf("\n");
} void catch_signal(int sign)
{
switch (sign)
{
case SIGINT:
printf("accept SIGINT!\n");
exit();
break;
case SIGQUIT:
printf("accept SIGQUIT!\n");
//取消信号阻塞
//定义信号集
sigset_t uset;
//清空信号集
sigemptyset(&uset);
//将SIGINT信号加入到信号集中
sigaddset(&uset,SIGINT);
//进行位异或操作,将信号集uset更新到进程控制块PCB结构中,取消阻塞信号SIGINT
sigprocmask(SIG_UNBLOCK,&uset,NULL);
break;
}
} int main(int arg, char *args[])
{
//定义未决信号集(pending)
sigset_t pset;
//定义阻塞信号集(block)
sigset_t bset;
//清空信号集
sigemptyset(&bset);
//将信号SIGINT加入到信号集中
sigaddset(&bset, SIGINT);
//注册信号
if (signal(SIGINT, catch_signal) == SIG_ERR)
{
perror("signal error");
return -;
}
if (signal(SIGQUIT, catch_signal) == SIG_ERR)
{
perror("signal error");
return -;
}
//进行位或操作,将信号集bset更新到进程控制块PCB结构中,阻塞信号SIGINT(即使用户按下ctrl+c,信号也不会递达)
sigprocmask(SIG_BLOCK, &bset, NULL);
while ()
{
/*
* 获取当前信号未决信息,即使在sigprocmask()函数中设置了信号阻塞,
* 但是如果没有信号的到来,信号未决状态字对应位依然是0
* 只要有信号到来,并且被阻塞了,信号未决状态字对应位才会是1
* */
sigpending(&pset);
//打印信号未决信息
printsigset(&pset);
sleep();
}
return ;
}

Linux 信号详解五(信号阻塞,信号未决)的更多相关文章
- Linux 信号详解六(可靠信号与不可靠信号)
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h&g ...
- 第三百五十五节,Python分布式爬虫打造搜索引擎Scrapy精讲—scrapy信号详解
第三百五十五节,Python分布式爬虫打造搜索引擎Scrapy精讲—scrapy信号详解 信号一般使用信号分发器dispatcher.connect(),来设置信号,和信号触发函数,当捕获到信号时执行 ...
- SIGPIPE信号详解
转自:http://blog.csdn.net/lmh12506/article/details/8457772 前一段面试的时候被问到项目中有没有处理SIGPIPE信号,怎么处理的?当时没有答出来, ...
- Linux 系统结构详解
Linux 系统结构详解 Linux系统一般有4个主要部分: 内核.shell.文件系统和应用程序.内核.shell和文件系统一起形成了基本的操作系统结构,它们使得用户可以运行程序.管理文件并使用系统 ...
- (转)Linux PS 详解
原文:https://cn.aliyun.com/jiaocheng/162702.html 摘要:原文地址:http://www.cnblogs.com/wangkangluo1/archive/2 ...
- Linux init详解(转)
Linux init详解 一.什么是INIT: init是Linux系统操作中不可缺少的程序之一. 所谓的init进程,它是一个由内核启动的用户级进程. 内核自行启动(已经被载入内存,开始运行,并已初 ...
- pika详解(五)登录认证及connectionParameters
pika详解(五)登录认证及connectionParameters 本文链接:https://blog.csdn.net/comprel/article/details/94662916 版权 pi ...
- Linux命令详解之—tail命令
tail命令也是一个非常常用的文件查看类的命令,今天就为大家介绍下Linux tail命令的用法. 更多Linux命令详情请看:Linux命令速查手册 Linux tail命令主要用来从指定点开始将文 ...
- Linux命令详解之—less命令
Linux下还有一个与more命令非常类似的命令--less命令,相比于more命令,less命令更加灵活强大一些,今天就给大家介绍下Linux下的less命令. 更多Linux命令详情请看:Linu ...
随机推荐
- XMLA连接器--免费但不开源通过ODBO、XMLA
XMLA 连接器(驱动) :来自ARQUERY http://jaist.dl.sourceforge.net/project/xmlaconnect/XMLA_Provider_v1.0.0.1 ...
- JSP中request getParameter和getAttribute不同(转载)
(1)request.getParameter()取得是通过容器的实现来取得通过类似post,get等方式传入的数据,,request.setAttribute()和getAttribute()只是在 ...
- 2.1 CMMI2级——7个PA简述
摘要: 阶段式的CMMI没有1级,最开始的级别就是2级.一个处于“无序化”生产的软件公司,要进行过程改进,首要是改进什么呢?2级告诉你,我们需要从计划.计划跟踪.需求管理.采购.度量.配置管理.质量保 ...
- javascript和web debug技术
在前端开发中,调试技术是必不可少的技能,本文将介绍五种前端开发必备的调试技术. Weinre移动调试 DOM 断点 debugger断点 native方法hook 远程映射本地调试 Weinre 在移 ...
- #一周五# win10通用平台,无处不在的Xamarin,msbuild开源,MVP卢建晖的Asp.NET 5系列 (视频)
又到周五,本周博主的大部分时间都花在深圳了.最近winhec的消息太多了,我只想补充一点,就是winhec时隔7年之后回归,大多数的媒体都还在沿用之前的“硬件工程大会(Hardware Enginee ...
- LoadRunner性能测试巧匠训练营
<LoadRunner性能测试巧匠训练营>基本信息作者: 赵强 邹伟伟 任健勇 丛书名: 实战出版社:机械工业出版社ISBN:9787111487005上架时间:2015-1-7出版日期: ...
- Mac上打开拷贝到移动硬盘里的文件提示“已经被osx使用不能打开”解决办法
在终端里面粘贴xattr -d com.apple.FinderInfo(此处按一下空格),然后把文件拖进终端窗口,按一下回车就好了
- 问题解决——开启Guest后仍无法共享打印机
==================================声明================================== 本文版权归作者所有 未经作者授权 请勿转载 保留法律追究的 ...
- MongoDB3.0.x版本用户授权配置(单机环境)
MongoDB数据库默认情况下是没有做权限控制的,只要能够连接所开放的端口就能进行访问,而且拥有root级别的权限:对于生产环境而言是极不安全的,所以需要建立用户,进行授权控制. 单机环境下的用户授权 ...
- Navicat for MySQL 工具注册码
在使用 MySQL 客户端中,感觉 Navicat for MySQL 做的还是很不错的.鄙人比较懒,喜欢采用绿色安装的方式,百度网盘上备份的有个绿色的压缩包,解压打开需要注册,这里记录一下注册码,版 ...