Linux —— 信号阻塞

目录

一,信号内核表示

sigset_t

sigprocmask

sigpending

二,捕捉信号

sigaction

三,可重入函数

四,volatile

五,SIGCHLD


信号常见概念

  • 实际执行信号的处理动作,称为信号递达Delivery;
  • 信号从产生到递达的状态,称为信号未决Pending;
  • 进程可选择阻塞某个信号;
  • 被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才会执行递达动作;
  • 阻塞和忽略是不同的,信号被阻塞就不会递达,忽略是递达后可选的一种处理动作;

一,信号内核表示

  • 每个信号都有两个标志:阻塞、未决,及一个函数指针表示的动作;信号产生时,内核在进程控制块中设置该信号的未决标志,直到信号递达才清除该标志;
    • SIGHUP信号,未产生也未阻塞,如递达时将执行默认动作;
    • SIGINT信号,产生过但被阻塞,暂时不能递达,处理动作为忽略;
    • SIGQUIT信号,未产生,如产生将被阻塞,处理动作为用户自定义函数;如该信号在阻塞前产生多次,POSIX允许系统递送该信号一次或多次,Linux常规信号在递达前产生多次只计一次,而实时信号在递达前产生多次可依次放在一个队列内;

sigset_t

  • 每个信号只有一个bit的未决标志,0或1,不记录该信号产生的次数;阻塞标志也是如此;
  • 未决和阻塞标志可用相同的数据类型sigset_t来存储,sigset_t称为信号集;该类型可表示每个信号的有效或无效;
//信号集操作函数
//在使用sigset_t类型的变量之前,一定要调用sigempty或sigfillset初始化,以使信号集处于确定状态;
//初始化后,即可调用sigaddset和sigdelset在信号集中添加或删除某种有效信号;
#include <signal.h>
int sigemptyset(sigset_t* set) //初始化信号集,使所有信号对应bit清零,表示该信号集不包含任何有效信号;
int sigfillset(sigset_t* set) //初始化信号集,使所有信号对应bit清零,表示该信号集的有效信号;
int sigaddset(sigset_t* set, int signo)
int sigdelset(sigset_t* set, int signo)
int sigismember(const sigset_t* set, int signo)

sigprocmask

  • 此函数可读取或更改进程的信号屏蔽字(阻塞信号集);
#include <signal.h>
int sigprocmask(int how, const sigset_t* set, sigset_t* oset);
  • 如oset是非空指针,则读取进程的当前信号屏蔽字通过oset参数传出;
  • 如set是非空指针,则更改进程的信号屏蔽字,参数how指示如何更改;
  • 如oset和set都是非空指针,则先将原来的信号屏蔽字备份到oset,然后根据set和how更改信号屏蔽字;
  • 如当前信号屏蔽字为mask,则下表说明了how参数的可选值;

  • 如调用此函数解除了对当前若干个未决信号的阻塞,则在sigprocmask返回前,至少将其中一个信号递达;

sigpending

  • 检测未决信号;
#include <signal.h>
int sigpending(sigset_t* set)
#include <iostream>    
#include <signal.h>    
#include <unistd.h>    
using namespace std;    void show_pending(sigset_t* pending){    for(int i=1; i<32; i++){    if(sigismember(pending, i))    cout<<"1";    else    cout<<"0";    }    cout<<endl;    
}    int main()    
{    sigset_t in, out;    sigemptyset(&in);    sigemptyset(&out);    sigaddset(&in, 2);    sigprocmask(SIG_SETMASK, &in, &out);    int count=0;    sigset_t pending;    while(1){    sigpending(&pending);    show_pending(&pending);    sleep(1);    if(count==10){    sigprocmask(SIG_SETMASK, &out, &in); //恢复2号信号后, 2信号立即递达并执行默认操作  cout<<"my: ";    show_pending(&in);    cout<<"recover default: ";    show_pending(&out);    }    count++;                                                                                                 }    return 0;    
} 
[wz@192 Desktop]$ g++ test.cpp -o test
[wz@192 Desktop]$ ./test 
0000000000000000000000000000000
0000000000000000000000000000000
0000000000000000000000000000000
^C0100000000000000000000000000000
0100000000000000000000000000000
0100000000000000000000000000000
0100000000000000000000000000000
0100000000000000000000000000000
0100000000000000000000000000000
0100000000000000000000000000000
0100000000000000000000000000000

二,捕捉信号

        如信号的处理动作是用户自定义函数,在信号递达时就调用该函数,称为捕捉信号;由于信号处理函数的代码在用户空间,处理过程比较复杂;如,用户程序注册了SIGQUIT信号的处理函数sighandler,当前正在执行main函数,此时发生中断或异常,切换达到内核态;在中断处理完毕后要返回用户态的main函数之前检查到有信号SIGQUIT递达;内核决定返回用户态后不是恢复main函数的上下文继续执行,而是执行sighandler函数,sighandler和main函数使用不同的堆栈空间,不存在调用和被调用的关系,是两个独立的控制流程;sighandler函数返回后自动执行特殊的系统调用sigreturn再次进入内核态;如没有新的信号递达,再返回用户态就是恢复main函数的上下文继续执行;

sigaction

        当某个信号的处理函数被调用时,内核自动将当前信号加入进程的信号屏蔽字,当信号处理函数返回时自动恢复原来的信号屏蔽字,这样就保证了在处理某个信号时,如这种信号再次产生,那么会被阻塞到当前处理结束为止;

        如在调用信号处理函数时,除了当前信号被自动屏蔽之外,还希望自动屏蔽另外一些信号,则用sa_mask字段说明这些需要额外屏蔽的信号,当信号处理函数返回时自动恢复原来的信号屏蔽字; 

#include <iostream>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
using namespace std;void handler(int signo){cout<<"get a signo: "<<signo<<endl;exit(10);
}int main(){struct sigaction act, oact;act.sa_handler = handler;act.sa_flags = 0;sigemptyset(&act.sa_mask);//act.sa_restorer = nullptr;//act.sa_sigaction = nullptr;sigaction(SIGINT, &act, &oact);while(1){cout<<"running..."<<endl;sleep(1);}return 0;
}

用户态,内核态;用户态需通过系统调用来访问内核数据,调用系统调用时系统会自动切换 身份;CPU会存在一个权限相关的寄存器数据,标识所处状态;每个用户进程都有自己的用户级页表,而OS只有一份内核页表;由于用户态和内核态的权限级别不同,所能看到的资源也是不一样的;

实时信号,不会丢失,会排队执行;

三,可重入函数

#include <iostream>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
using namespace std;void show(int signo){int i=0;while(i<5){cout<<"show(), signo: "<<signo<<endl;i++;sleep(1);}
}void handler(int signo){cout<<"handler calling..."<<endl;show(signo);
}int main(){struct sigaction act, oact;act.sa_handler = handler;act.sa_flags = 0;sigemptyset(&act.sa_mask);sigaction(SIGINT, &act, &oact);show(999);return 0;
}
[wz@192 Desktop]$ ./test
show(), signo: 999
show(), signo: 999
show(), signo: 999
^Chandler calling...
show(), signo: 2
show(), signo: 2
show(), signo: 2
show(), signo: 2
show(), signo: 2
show(), signo: 999
show(), signo: 999

        像以上,insert插入函数被不同控制流调用,可能在第一次调用还没返回时,就再次进入该函数,称为重入;insert函数访问一个全局链表,有可能因为插入而造成错乱,像这样的函数称为不可重入函数;反之,如一函数只访问自己的局部变量或参数,称为可重入函数;所学的大部分函数都是不可重入的;

如函数符合以下条件之一,则是不可重入:

  • 调用了malloc或free,因malloc也是也是用全局链表来管理堆的;
  • 调用了标准I/O函数,标准I/O库的很多实现都以不可重入的方式使用全局数据结构;

四,volatile

        C语言关键字,保持内存的可见性;

#include <stdio.h>
#include <signal.h>int flag=0;
void handler(int signo){flag=1;printf("handler calling, get signo: %d\n", signo);
}int main(){signal(2, handler);while(!flag); //注意没有循环体printf("process quit normal!\n");return 0;
}
[wz@192 Desktop]$ gcc -o test test.c 
[wz@192 Desktop]$ ./test
^Chandler calling, get signo: 2
process quit normal!
//优化级别1
[wz@192 Desktop]$ gcc -o test test.c -O1
[wz@192 Desktop]$ ./test
^Chandler calling, get signo: 2
^Chandler calling, get signo: 2
^Chandler calling, get signo: 2

        优化后,flag被放在了CPU的寄存器当中,while循环的flag并不是内存中的最新flag;使用volatile关键字修饰变量后,则该变量不允许在被优化,对该该变量的任何操作都必须在真实的内存中进行;

#include <stdio.h>
#include <signal.h>volatile int flag=0;
void handler(int signo){flag=1;printf("handler calling, get signo: %d\n", signo);
}int main(){signal(2, handler);while(!flag); //注意没有循环体printf("process quit normal!\n");return 0;
}
[wz@192 Desktop]$ gcc -o test test.c -O1
[wz@192 Desktop]$ ./test
^Chandler calling, get signo: 2
process quit normal!

五,SIGCHLD

        SIGCHLD是第17号信号;进程wait、waitpid函数清理僵死进程,父进程可阻塞等待子进程结束,也可非阻塞查询是否有子进程结束等待清理(轮询);第一种方式父进程阻塞了,就不能处理自己的工作,第二种方式父进程在处理自己的工作同时还要记得轮询,程序实现复杂;

        其实,子进程在终止时会给父进程发送SIGCHLD信号,该信号默认处理动作为忽略,父进程可自定义SIGCHLD信号的处理函数;这样父进程只需专心处理自己的工作,不必关心子进程;子进程终止时通知父进程,父进程在信号处理函数中调用wait清理子进程即可;

        由于UNIX的历史原因,要想不产生僵死进程,还可在父进程调用sigaction时将SIGCHLD处理动作置为SIG_IGN,这样fork出来的子进程在终止时会自动清理,不会产生僵死进程,也不会通知父进程;系统默认的忽略动作和用户用sigaction函数自定义的忽略,通常是没有区别的,但这是特例;此方法对于Linux可用,但不保证在其他UNIX系统上都可使用;

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
#include <signal.h>void handler(int signo){printf("father get signo: %d\n", signo);pid_t id;//可能有多个子进程while((id=waitpid(-1,NULL,WNOHANG))>0){printf("wait child success: %d\n", id);}printf("child is quit! %d\n", getpid());
}int main(){signal(SIGCHLD, handler);pid_t cid;if((cid=fork()) == 0){printf("child: %d\n", getpid());sleep(3);exit(1);}while(1){printf("father process...!\n");sleep(1); //可以提前被唤醒}return 0;
}
[wz@192 Desktop]$ gcc -o test test.c
[wz@192 Desktop]$ ./test
father process...!
child: 99919
father process...!
father process...!
father get signo: 17
wait child success: 99919
child is quit! 99918
father process...!
father process...!
father process...!
father process...!

如不设置signal,子进程终止时,就会产生僵死进程;

如设置为SIG_IGN,子进程终止时,自动清理;

//如设置为忽略,fork出来的子进程在终止时会自动清理,不会产生僵死进程
signal(SIGCHLD, SIG_IGN);

        等待子进程,避免Z进程内存泄露,可能需获取子进程的退出码;父进程不关心子进程退出码,可不wait,如关心退出码必须wait;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/128392.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深眸科技自研轻辙视觉引擎,以AI机器视觉赋能杆号牌识别与分拣

电线杆号牌作为电力行业标识的一种&#xff0c;相当于电线杆的“身份证”&#xff0c;担负着宣传电力知识、安全警示的作用&#xff0c;用于户外使用标记输电线路电压等级、线路名称、杆塔编号等&#xff0c;能够清晰地记录电力线路杆的信息&#xff0c;并为电力线路的更改以及…

面试问题总结(1)

(꒪ꇴ꒪ )&#xff0c;Hello我是祐言QAQ我的博客主页&#xff1a;C/C语言&#xff0c;数据结构&#xff0c;Linux基础&#xff0c;ARM开发板&#xff0c;网络编程等领域UP&#x1f30d;快上&#x1f698;&#xff0c;一起学习&#xff0c;让我们成为一个强大的攻城狮&#xff0…

Matlab 如何计算正弦信号的幅值和初始相角

Matlab 如何计算正弦信号的幅值和初始相角 1、概述 如果已知一个正弦信号的幅值&#xff0c;在FFT后频域上该信号谱线的幅值与设置值不同&#xff0c;而是大了许多&#xff1b;如果不知道某一正弦信号的幅値&#xff0c;又如何通FFT后在頻域上求出该正弦信号的幅值呢? 2、…

Python 交易指南:利用 RSI

一、说明 RSI是相对强弱指数&#xff08;Relative Strength Index&#xff09;的缩写&#xff0c;是一种技术指标。该指标是用来测量股票或其他交易品种的价格波动强度和速度的&#xff0c;属于动量型指标。RSI常用于技术分析和交易策略中&#xff0c;可以帮助交易者判断市场的…

C语言:三子棋小游戏

简介&#xff1a; 目标很简单&#xff1a;实现一个 三子棋小游戏。三子棋大家都玩过&#xff0c;规则就不提及了。本博文中实现的三子棋在对局中&#xff0c;电脑落子是随机的&#xff0c;不具有智能性&#xff0c;玩家的落子位置使用键盘输入坐标。下面开始详细介绍如何实现一…

QT实战之翻金币游戏【未完待续】

文章目录 目录 文章目录 前言 二、创建项目 三、添加资源 四、主界面实现 1、设置游戏主场景配置 2、设置背景图片 3、创建开始按钮 总结 前言 对QT的相关知识与控件进行简单的学习之后&#xff0c;通过实现“翻金币游戏”来巩固与实践所学的QT知识。在制作过程中是根据以下视…

PHP8数组的类型-PHP8知识详解

php 8 引入了对数组的类型提示&#xff0c;以帮助开发者更准确地定义和验证数组的结构。以下是 PHP 8 中支持的数组类型&#xff1a;索引数组、关联数组、混合类型数组。 1、索引数组 (Indexed arrays): PHP索引数组一般表示数组元素在数组中的位置&#xff0c;它由数字组成&a…

飞行动力学 - 第18节-part2-航向操纵面 之 基础点摘要

飞行动力学 - 第18节-part2-航向操纵面 之 基础点摘要 1. 航向操纵面2. 非常规航向操纵面3. 正方向舵偏角产生的偏航力矩4. 产生或平衡侧滑角 β \beta β所需的方向舵偏角5. 参考资料 1. 航向操纵面 方向舵是航向的主要操纵面。 2. 非常规航向操纵面 开裂式阻力方向舵 ( Spl…

el-table操作列动态自适应设置(根据操作项个数动态设置宽度)

一、目的 目的&#xff1a;表格操作列宽度&#xff0c;根据操作项多少&#xff0c;自动调节宽度背景&#xff1a;用el-table组件开发时&#xff0c;对于表格的操作列的自适应宽度是一个问题&#xff0c;如果不设置&#xff0c;操作按钮多时会有换行问题。如果设置最小宽度或宽…

跟踪源码技巧+阅读源码工具Sourcetrail

基于Eclipse IDE 1、Quick Type Hierarchy 快速查看类继承体系 &#xff08; 快捷键&#xff1a;Ctrl T&#xff09; 查看类很多人可能都知道&#xff0c;可源码阅读的时候更多用来查看方法体系更重要&#xff0c;可以方便快速的定位到方法的实现类。如&#xff1a; 此时如…

使用docker搭建owncloud Harbor 构建镜像

1、使用mysql:5.6和 owncloud 镜像&#xff0c;构建一个个人网盘。 2、安装搭建私有仓库 Harbor 3、编写Dockerfile制作Web应用系统nginx镜像&#xff0c;生成镜像nginx:v1.1&#xff0c;并推送其到私有仓库。具体要求如下&#xff1a; &#xff08;1&#xff09;基于centos基础…

SpringSecurity学习

1.认证 密码校验用户 密码加密存储 Configuration public class SecurityConfig extends WebSecurityConfigurerAdapter {Beanpublic PasswordEncoder passwordEncoder(){return new BCryptPasswordEncoder();}} 我们没有以上代码配置&#xff0c;默认明文存储, {id}password…

解决eNSP和HCL Cloud兼容性的问题

问题&#xff1a;eNSP或HCL无法启动 不兼容的原因&#xff1a;eNSP支持Virtual Box是5.2.44&#xff1b;HCL支持的Virtual Box版本是6.0.14 解决方案&#xff1a;注册表欺骗 再进行重新安装前先把之前的都卸载掉&#xff1a;eNSP、VirtualBox、HCL等 1、先安装Virtual Box 5.…

23个react常见问题

1、setState 是异步还是同步&#xff1f; 合成事件中是异步 钩子函数中的是异步 原生事件中是同步 setTimeout中是同步 相关链接&#xff1a;你真的理解setState吗&#xff1f;&#xff1a; 2、聊聊 react16.4 的生命周期 图片 相关连接&#xff1a;React 生命周期 我对 Reac…

【数据结构】栈---C语言版(详解!!!)

文章目录 &#x1f438;一、栈的概念及结构&#x1f344;1、栈的概念定义&#x1f344;2、动图演示&#x1f332;入栈&#x1f332;出栈&#x1f332;整体过程 &#x1f438;二、栈的实现&#x1f438;三、数组结构栈详解&#x1f34e;创建栈的结构⭕接口1&#xff1a;定义结构…

【uniapp/uview】u-datetime-picker 选择器的过滤器用法

引入&#xff1a;要求日期选择的下拉框在分钟显示时&#xff0c;只显示 0 和 30 分钟&#xff1b; <u-datetime-picker :show"dateShow" :filter"timeFilter" confirm"selDateConfirm" cancel"dateCancel" v-model"value1&qu…

yolov5运行过程遇到的小问题(随时更新)

1.关于git的问题 解决办法&#xff1a;插入下面代码 import os os.environ["GIT_PYTHON_REFRESH"] "quiet"2.页面太小无法完成操作 解决办法: 如果不好使再考虑降低Batch_Size大小或者调整虚拟内存可用硬盘空间大小&#xff01;&#xff08;调整虚拟内存…

实现无公网IP的公网环境下Windows远程桌面Ubuntu 18.04连接,高效远程办公!

文章目录 一、 同个局域网内远程桌面Ubuntu1. 更新软件仓库2. 安装支持包3. 安装XFCE4桌面环境4. 安装XRDP5. 环境设置5.1 XFCE桌面配置5.2 在配置文件中&#xff0c;加入XFCE会话 6 重启服务7. 查看IP地址8. 使用Windows远程桌面连接 二、公网环境系统远程桌面Ubuntu1. 注册cp…

【Java】Jxls--轻松生成 Excel

1、介绍 Jxls 是一个小型 Java 库&#xff0c;可以轻松生成 Excel 报告。Jxls 在 Excel 模板中使用特殊标记来定义输出格式和数据布局。 Java 有一些用于创建 Excel 文件的库&#xff0c;例如Apache POI。这些库都很好&#xff0c;但都是一些较底层的库&#xff0c;因为它们要…

【网络安全】图解 Kerberos:身份认证

图解 Kerberos&#xff1a;身份认证 1.什么是 Kerberos &#xff1f;2.Kerberos 基本概念2.1 基本概念2.2 KDC 3.Kerberos 原理3.1 客户端与 Authentication Service3.2 客户端与 Ticket Granting Service3.3 客户端与 HTTP Service Kerberos 是一种身份认证协议&#xff0c;被…