【Linux】学习-进程信号

进程信号

信号入门

生活角度的信号

  • 你在网上买了很多件商品,再等待不同商品快递的到来。但即便快递没有到来,你也知道快递来临时,你该怎么处理快递。也就是你能“识别快递”,也就是你意识里是知道如果这时候快递员送来了你的包裹,你知道该如何处理这些包裹
  • 当快递员到了你楼下,你也收到快递到来的通知,但是你正在打游戏,需5min之后才能去取快递。那么在在这5min之内,你并没有下去去取快递,但是你是知道有快递到来了。也就是取快递的行为并不是一定要立即执行,可以理解成“在合适的时候去取”。你可以暂时将包裹搁置,等到有空的时候再处理。
  • 在收到通知,再到你拿到快递期间,是有一个时间窗口的,在这段时间,你并没有拿到快递,但是你知道有一个快递已经来了。本质上是你“记住了有一个快递要去取”
  • 当你时间合适,顺利拿到快递之后,就要开始处理快递了。而处理快递一般方式有三种:1. 执行默认动作(幸福的打开快递,使用商品)2. 执行自定义动作(快递是零食,你要送给你你的女朋友)3. 忽略快递(快递拿上来之后,扔掉床头,继续开一把游戏)
  • 快递到来的整个过程,对你来讲是异步的,你不能准确断定快递员什么时候给你打电话

技术应用角度的信号

什么是Linux信号?

  • 信号是进程之间事件异步通知的一种方式,属于软中断

  • 信号本质是一种通知机制,用户or操作系统通过发送一定的信号,通知进程,某些事件已经发生,进程可以根据信号的种类来进行相关处理、

结合生活信号,对信号的一些小结论

  • 进程要处理信号,必须具备信号的“识别”能力,和此能力是程序员给进程植入的,可以理解成出生就自带的
  • 信号种类繁多,因此进程收到信号后,有可能是立即执行的,也有可能是后续处理的
  • 信号能够被进程所保存,因此也就具备后续处理的能力
  • 一般而言,信号的产生相对于进程而言是异步的,也就是说,信号发送的时候,进程是正在干自己的事情的,随时都有可能收到信号,具备识别能力但不代表提前得知

先简单理解一下信号被进程保存

  • 进程内部有相关的数据结构位图,信号每次被发送给进程时,此时信号处于未决状态,那么用来保存信号的字段表中信号编号对应的位置就会由0变1

    信号处理的三种方式

  • 默认处理

  • 忽略

  • 自定义捕捉

产生信号

通过终端按键产生信号

  • 我们平常在写代码时,其实经常都有在跟信号接触,看以下代码:

    #include <stdio.h>
    #include <unistd.h>
    int main()
    {while(1){printf("I am a process, I am waiting signal!\n");sleep(1);}
    }
    

    image-20231008160048771

由于没有退出条件,因此进程一直处于死循环,此时我们在键盘里键入:Ctrl+c 能够将进程终止,其实按下组合键后本质就是向当前在前台运行的进程发送了终止信号,进程也立马对此信号做出了反应,终止掉了进程。这就是产生信号的其中一种方式

常见信号

  • 信号一共有61个信号,32,33,0号信号是不存在的,而1-31的信号称为普通信号,32-64为实时信号,我们只学习普通信号:

    image-20231008160757094

  • 关于信号的详细信息我们可以用man手册查看:man 7 signal

    image-20231008160703067

  • 每个信号的编号都是被宏定义好的:

    /* Signals.  */
    #define	SIGHUP		1	/* Hangup (POSIX).  */
    #define	SIGINT		2	/* Interrupt (ANSI).  */
    #define	SIGQUIT		3	/* Quit (POSIX).  */
    #define	SIGILL		4	/* Illegal instruction (ANSI).  */
    #define	SIGTRAP		5	/* Trace trap (POSIX).  */
    #define	SIGABRT		6	/* Abort (ANSI).  */
    #define	SIGIOT		6	/* IOT trap (4.2 BSD).  */
    #define	SIGBUS		7	/* BUS error (4.2 BSD).  */
    #define	SIGFPE		8	/* Floating-point exception (ANSI).  */
    #define	SIGKILL		9	/* Kill, unblockable (POSIX).  */
    #define	SIGUSR1		10	/* User-defined signal 1 (POSIX).  */
    #define	SIGSEGV		11	/* Segmentation violation (ANSI).  */
    #define	SIGUSR2		12	/* User-defined signal 2 (POSIX).  */
    #define	SIGPIPE		13	/* Broken pipe (POSIX).  */
    #define	SIGALRM		14	/* Alarm clock (POSIX).  */
    #define	SIGTERM		15	/* Termination (ANSI).  */
    #define	SIGSTKFLT	16	/* Stack fault.  */
    #define	SIGCLD		SIGCHLD	/* Same as SIGCHLD (System V).  */
    #define	SIGCHLD		17	/* Child status has changed (POSIX).  */
    #define	SIGCONT		18	/* Continue (POSIX).  */
    #define	SIGSTOP		19	/* Stop, unblockable (POSIX).  */
    #define	SIGTSTP		20	/* Keyboard stop (POSIX).  */
    #define	SIGTTIN		21	/* Background read from tty (POSIX).  */
    #define	SIGTTOU		22	/* Background write to tty (POSIX).  */
    #define	SIGURG		23	/* Urgent condition on socket (4.2 BSD).  */
    #define	SIGXCPU		24	/* CPU limit exceeded (4.2 BSD).  */
    #define	SIGXFSZ		25	/* File size limit exceeded (4.2 BSD).  */
    #define	SIGVTALRM	26	/* Virtual alarm clock (4.2 BSD).  */
    #define	SIGPROF		27	/* Profiling alarm clock (4.2 BSD).  */
    #define	SIGWINCH	28	/* Window size change (4.3 BSD, Sun).  */
    #define	SIGPOLL		SIGIO	/* Pollable event occurred (System V).  */
    #define	SIGIO		29	/* I/O now possible (4.2 BSD).  */
    #define	SIGPWR		30	/* Power failure restart (System V).  */
    #define SIGSYS		31	/* Bad system call.  */
    #define SIGUNUSED	31
    
  • 其中,Ctrl+c 组合键对应的就是2号信号SIGINT,它所对应的Action行为是Term行为:Terminate终止动作

注意

  1. Ctrl-C 产生的信号只能发给前台进程。一个命令后面加个&可以放到后台运行,这样Shell不必等待进程结束就可以接受新的命令,启动新的进程。
  2. Shell可以同时运行一个前台进程和任意多个后台进程,只有前台进程才能接到像 Ctrl-C 这种控制键产生的信号。
  3. 前台进程在运行过程中用户随时可能按下 Ctrl-C 而产生一个信号,也就是说该进程的用户空间代码执行到任何地方都有可能收到 SIGINT 信号而终止,所以信号相对于进程的控制流程来说是异步(Asynchronous)的。

如何理解组合键变信号?

  • 键盘的工作方式是通过中断方式进行的,因此操作系统能读取并解析组合键,解析完毕后会查找进程列表中在前台运行的进程,并把对于的信号写入进程!

初始信号捕捉

我们前面提到过,信号被写入进程后,进程对此信号的处理有三种方式,我们先来简单介绍一下自定义捕捉方式

介绍signal函数

  • image-20231008165135410

    功能:能够捕捉指定的信号编号为signum的信号,然后将此信号的处理方式使用我们自定义的方式handler

  • 其中:sighandler_t是一个函数指针,以一个的函数指针作为另一个函数的参数并在函数内使用函数指针调用对应指向的函数,此函数称为回调函数,自定义捕捉时,我们实现自定义函数,并将函数作为参数传给signal,通过回调的方式,来修改对应信号的捕捉方法。返回值是自定义捕捉前的函数的地址。

通过信号捕捉的方式,验证Ctrl+c组合键是SIGINT信号

  • #include <iostream>
    #include <unistd.h>
    #include <signal.h>
    using namespace std;
    void cathSig(int signum)
    {cout<<"i catch you! SIGINT!"<<endl;
    }
    int main()
    {signal(SIGINT,cathSig);while(1){cout<<"my pid:"<<getpid()<<endl;sleep(1);}
    }
    

image-20231008171152188

我们可以观察到,按下Ctrl-c后进程并不会像之前一样退出,而是执行了我们自己的代码!得以验证以下两个结论:

  • Ctrl-c对应的信号为SIGINT信号
  • Ctrl-c对应的信号处理方式为终止进程,自定义捕捉方式后,原先的处理方法就被舍弃了

注意

  • signal函数一般写在最前面,原理类似于我们要先买票,才能对信号进行自定义捕捉,后面讲阻塞信号时会解释原理!

Core Dump

我们先来比较两个信号

  • image-20231008171758567

    这两个信号都是从键盘敲组合键来向进程发送信号,其中:

    • Ctrl-c 对应的是2号SIGINT
    • Ctrl-\ 对应的是3号信号SIGQUIT
  • 他们两个的作用都能够用来终止进程,但不同的是:

    2号对应的行为是Term:Terminate

    3号对应的行为是Core:Core Dump

  • 简单验证:image-20231008172753177

什么是Core Dump?

  • Core Dump,又叫核心转储,当一个进程要异常终止时,可以选择把进程的用户空间内存数据全部 保存到磁盘上,文件名通常是core,这叫做Core Dump。进程异常终止通常是因为有Bug,比如非法内存访问导致段错误,事后可以用调试器检查core文件以查清错误原因,这叫做Post-mortem Debug(事后调试)。一个进程允许产生多大的core文件取决于进程的Resource Limit(这个信息保存 在PCB中)。
  • 默认是不允许产生core文件的(一般而言,云服务器-生产环境下,核心转储的功能是被关闭的!),因为core文件中可能包含用户密码等敏感信息,不安全。
  • 在开发调试阶段可以用ulimit命令改变这个限制,允许产生core文件。 首先用ulimit命令改变Shell进程的Resource Limit,允许core文件最大为1024K: $ ulimit -c1024

使用命令ulimit命令查看并修改相关资源配置

  • image-20231008173314127

显示0时,代表核心转储不被允许产生,我们需要自行开启:

image-20231008173420309

此时已显示core文件允许的最大内存为1024K,但仅仅在当前会话生效,退出会话后就失效了

  • ulimit命令改变了Shell进程的Resource Limit,test进程的PCB由Shell进程复制而来,所以也具 有和Shell进程相同的Resource Limit值,这样就可以产生Core Dump了。

验证Ctrl-\ 发送的SIGQUIT能够产生core文件的功能

  • 先写一个死循环程序:

    int main()
    {
    while(1)
    {printf("my pid:%d\n",getpid());sleep(1);
    }
    }
    

    运行并用Ctrl-c命令终止后发现并没有产生任何文件:

    image-20231009100614215

  • 再次运行并使用Ctrl-\ 命令终止:

    image-20231009101004912

    这次我们发现产生了core.27213文件,并且后缀就是以进程的pid来命名的,代表此文件就是此进程出现某种异常时,由os将此进程在内核中相关的核心数据转存到磁盘中

核心转储有什么用?

  • 主要是用来进行调试,有了核心转储的数据,调试时会方便很多

利用核心转储进行调试

  • 除了SIGQUIT信号的行为是core,还有其他很多命令也有core行为,下面我们使用SIGFPE信号来验证使用核心转储调试时是否会方便image-20231009103130248

    FPE:Floating point exception指的是浮点数错误,一般指除0错误

  • 首先编写程序,并且编译时,带上-g选项,生成debug文件

    int main()
    {while (1){printf("my pid:%d\n", getpid());sleep(1);int a=100;a/=0;printf("run here...\n");}
    }
    

    image-20231009103412230

  • 重新运行程序,此时也随之生成了core dump文件

    image-20231009103454313

  • 启用gdb调试程序,输入core-file core.pid(core文件名)

    image-20231009103654527

此时我们能发现,在core dump文件的帮助下,能自动帮我们定位在哪一行代码收到了什么信号,比如这里,我们在29行除0错误处收到了8号信号,而8号信号就是对应的SIGFPE信号

core dump标记位

  • 其实我们在进程间控制时就已经和Core Dump有过一面之缘了,我们来看这张图:

在进程控制一篇提到过,子进程退出时,父进程可以通过进程等待的方式,收集子进程的退出信息,以防子进程变成僵尸进程,其中进程等待可以使用wait和waitpid系统调用image-20231009104433908

其中我们也提到过status参数是一个输出型参数,其中如果子进程是被信号所杀的话,此参数的低七位会被填入终止信号的编号,而第八位则是显示是否有生成core dump文件,我们也可以通过代码来验证一下:

用子进程验证core dump标记位

  • int main()
    {pid_t id = fork();if (id == 0){// childsleep(1);int a = 100;a /= 0;exit(0);}int status = 0;waitpid(id, &status, 0);cout << "father:" << getpid()<<" " << "child:" << id <<" "<< "exit sig:" 

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

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

相关文章

软考 系统分析师系列知识点之信息系统战略规划方法(5)

接前一篇文章&#xff1a;软考 系统分析师系列知识点之信息系统战略规划方法&#xff08;4&#xff09; 所属章节&#xff1a; 第7章. 企业信息化战略与实施 第4节. 信息系统战略规划方法 7.4.3 战略集合转化法 战略目标集合转化法&#xff08;Strategy Set Transformation&a…

day37 闭包、变量提升

目录 闭包变量提升函数提升 闭包 闭包&#xff08;closure&#xff09;是一个函数以及其捆绑的周边环境状态&#xff08;lexical environment&#xff0c;词法环境&#xff09;的引用的组合。换而言之&#xff0c;闭包让开发者可以从内部函数访问外部函数的作用域。在 JavaScr…

Oracle数据字典学习1

之前查看了几个用户的默认表空间&#xff0c;是从user_users来查看的&#xff1b; 根据资料&#xff1b; ORACLE中数据字典视图分为3大类,用前缀区别&#xff0c;分别为&#xff1a;USER&#xff0c;ALL 和 DBA&#xff1b; 许多数据字典视图包含相似的信息&#xff1b; USER_…

【PyQt】10 QLineEdit

文章目录 前言一、回显模式&#xff08;EchoMode&#xff09;1.1 四种回显模式1.2 代码展示运行结果 二、校验器2.1 代码2.2 运行结果 三、通过掩码限制输入3.1 代码3.2 运行结果 总结 前言 1、QLineEdit 可以输入单行文字 2、回显模式 3、校验器 4、掩码输入 一、回显模式&am…

Blazor SSR/WASM IDS/OIDC 单点登录授权实例3-服务端管理组件

目录: OpenID 与 OAuth2 基础知识Blazor wasm Google 登录Blazor wasm Gitee 码云登录Blazor SSR/WASM IDS/OIDC 单点登录授权实例1-建立和配置IDS身份验证服务Blazor SSR/WASM IDS/OIDC 单点登录授权实例2-登录信息组件wasmBlazor SSR/WASM IDS/OIDC 单点登录授权实例3-服务端…

黄金交易策略(Nerve Nnife.mql4):1秒救地球的第六单

一轮趋势做单&#xff0c;正常情况是5单便可以完成一轮盈利。但当开仓后快速追加5单也无法止盈的话&#xff0c;我们得找准极其苛刻的条件开出第6单&#xff0c;并指望完成利润覆盖。代码如下&#xff1a; if(count > 5 && count < 10 && isDown(small_…

《CSS 简易速速上手小册》第9章:CSS 最佳实践(2024 最新版)

文章目录 9.1 维护大型项目的 CSS9.1.1 基础知识9.1.2 重点案例&#xff1a;构建一个可复用的 UI 组件库9.1.3 拓展案例 1&#xff1a;优化现有项目的 CSS 结构9.1.4 拓展案例 2&#xff1a;实现主题切换功能 9.2 BEM、OOCSS 和 SMACSS 方法论9.2.1 基础知识9.2.2 重点案例&…

推荐系统|召回04_离散特征处理

离散特征处理 离散特征是什么 怎么处理离散特征 One-hot编码 Embedding嵌入 从one-hot到Embedding&#xff0c;已经节省了很多的存储空间&#xff0c;但当数据量大的时候&#xff0c;还是占空间&#xff0c;所以工业界仍会对Embedding进行优化 而一个物品所对应的Embedding参数…

ESP8266-01S模块连接服务器(ONENET)

ESP8266-01S模块连接服务器&#xff08;ONENET&#xff09; 固件烧录通过http协议连接onenet云平台 固件烧录 一般在esp8266到手之后会进行测试&#xff0c;以确保能用&#xff0c;该模块出厂时默认自带出厂固件的&#xff0c;但如果进行了开发即下载了自己写的程序&#xff0…

Java:常用API接上篇 --黑马笔记

一、 StringBuilder类 StringBuilder代表可变字符串对象&#xff0c;相当于是一个容器&#xff0c;它里面的字符串是可以改变的&#xff0c;就是用来操作字符串的。 好处&#xff1a;StringBuilder比String更合适做字符串的修改操作&#xff0c;效率更高&#xff0c;代码也更…

MySQL-SQL优化

文章目录 1. SQL性能分析1.1 SQL执行频率1.2 慢查询日志1.3 profile详情1.4 explain 2. SQL优化2.1 Insert 优化2.2 Group By 优化2.3 Order By 优化2.4 Limit 优化2.5 Count() 优化2.6 Update 优化 3. 拓展3.1 请你说一下MySQL中的性能调优的方法&#xff1f;3.2 执行 SQL 响应…

STM32 新建寄存器版本MDK工程简要步骤

新建工程文件夹 新建一个工程根目录文件夹&#xff0c;并在该文件夹里新建D/M/O/P/U文件夹。 Drivers&#xff1a;存放与硬件相关的驱动层文件Middlewares&#xff1a;存放正点原子提供的中间层组件文件和第三方中间层文件Output&#xff1a;存放工程编译输出文件Projects&am…