科林Linux_4 信号

#include <signal.h>

信号signal:Linux或Unix系统支持的经典的消息机制,用于处置进程,挂起进程或杀死进程

kill -l    #查看系统支持的信号

1~31 Unix经典信号(软件开发工程师)

32、33信号被系统隐藏,不对用户开放,供NPTL线程库使用

34~64 自定义信号/实时信号(驱动工程师)

Ctrl+C 系统帮你发送了 2)SIGINT信号。终端组合按键产生的信号,杀死唯一的前台终端进程。

一、系统中触发信号的几种方式:

1、终端组合按键触发信号

Ctrl+C SIGINT/2 终止进程        Ctrl+\ SIGQUIT/3 退出进程        Ctrl+Z SIGTSTP/20 挂起终端进程

jobs    #查看挂起的作业编号
fg 作业编号    #唤醒到前台

tty 终端 pts 虚拟终端

               D (TASK_UNINTERRUPTIBLE)     不可中断的睡眠状态
               R (TASK_RUNNING)                正在运行,或在队列中的进程
               S (TASK_INTERRUPTIBLE)        可中断的睡眠状态
               T (TASK_STOPPED)                停止状态
               t (TASK_TRACED)                被跟踪状态
               Z (TASK_DEAD - EXIT_ZOMBIE)  退出状态,但没被父进程收尸,成为僵尸状态
               W                            进入内存交换(从内核2.6开始无效)
               X (TASK_DEAD - EXIT_DEAD)    退出状态,进程即将被销毁


               <    高优先级
               N    低优先级
               L    有些页被锁进内存
               s    包含子进程
               +    位于前台的进程组;
               l    多线程,克隆线程  multi-threaded (using CLONE_THREAD, like NPTL pthreads do)

 2、命令触发信号

kill -signo信号 pid进程号    #向任意进程发送任意信号

3、函数触发信号

kill(pid_t pid,int signo);    //向任意进程发送任意信号
raise(int signo);    //向当前进程发送任意信号
abort();    //向当前进程发送固定的SIGABRT/6信号
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>

int main(int argc,char* argv[]){
	//支持命令,KILL signo pid
	if(argc<3){
		printf("pram error\n");
		exit(0);
	}
	kill(atoi(argv[2]),atoi(argv[1]));
	return 0;
}

4、硬件异常产生信号

违规访问使用硬件,导致信号杀死进程

1)如果进程非法操作内存,系统向其发送SIGSEGV/11杀死进程,错误信号为段错误

    char* str="hello";
    str[0]='H';    //只读内存的写操作

 2)如果进程出现cpu运算异常,系统向其发送SIGFPE/8杀死进程,错误为浮点数例外

    int b=0;    
    int a=8/b;    //除0

3)如果进程出现内存访问越界,系统向其发送SIGBUS/7杀死进程,错误为总线错误

    int* ptr=NULL;
	int fd=open("MapFile",O_RDWR);
	int size=lseek(fd,0,SEEK_END);
    ptr=mmap(NULL,size,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);    
    //在使用mmap内存映射时,若"MapFile"是内容为空的新文件大小为0,没有与之对应的合法的物理页,mmap不能扩展,发生了越界访问
    //添加错误检测来防止总线错误
    if((ptr=mmap(NULL,size,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0))==MAP_FAILED){//内存映射
		perror("mmap failed");
		exit(0);
	}

5、软件异常产生信号

可以使用某个组件,但是如果使用时触发软条件,系统会杀死进程(信号)

1)定时器alarm(10),定时到时,系统向定时进程发送SIGALRM/14信号,默认杀死进程

    alarm(1);    //定时1s后杀死进程,返回“闹钟”

2)匿名管道读端关闭,写端向管道写数据,系统向写端发送SIGPIPE/13信号,杀死进程

二、信号的三大行为与处理动作:

信号的行为可以被修改,默认情况下为默认行为,但是可以改为忽略或捕捉(三选一)

1、默认行为SIG_DFL:默认五种处理动作(五选一)

(1)TERM 直接杀死进程(2)CORE 杀死进程并且转储核心(3)STOP 挂起进程(4)CONT 忽略/继续进程(5)IGN 忽略

每个信号都有自己的默认动作来处置进程,进程不是被挂起就是被杀死。只有动作为IGN的信号,不会处置进程。

可以通过信号结束进程的输出,分析其动作

2、忽略行为SIG_IGN:忽略行为没有处理动作,无法处置进程

3、捕捉行为SIG_ACTION:可以手写捕捉函数来自定义行为

void sig_do(int n);

信号捕捉可以让自定义的捕捉函数与信号绑定,以后触发此信号,系统就会去执行此函数 

可以用来设计实现条件触发的工作和任务,信号触发则执行,否则不执行。

修改信号行为

每个信号都有一个自己的信号行为结构体,修改信号行为要保留其原结构体,便于复位

    struct sigaction nact,oact;    //信号行为结构体
    act.sa_handler=SIG_DFL|SIG_IGN|函数指针;
    act.sa_flags=0;    //默认选项
    act.sa_mask;    //临时屏蔽字

    sigaciton(signo,&nact,&oact);    //替换信号行为结构体,act为新的,oact传出进程原有的
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>

void sig_do(int n){
	printf("已经成功捕捉SIGINT,signo=%d\n",n);
}

int main(){
	//void(*sa_handle)(int)
	struct sigaction act,oact;
	act.sa_handler=sig_do;
	act.sa_flags=0;
	sigemptyset(&act.sa_mask);//初始化
	sigaction(SIGINT,&act,&oact);//替换信号行为
	while(1)
		sleep(1);
	return 0;
}

三、三种让信号生效的方法:

1、信号忽略,将信号设置为忽略行为

2、信号捕捉,对信号进行捕捉设定,绑定捕捉函数

3、信号屏蔽:阻塞信号的传递

如果信号通过未决信号集,系统会将未决中对应的位设置为1,标记未决同时避免相同的信号同时处理。

如果未决为1,对应的信号直接丢弃,不处理

信号的处理不支持排队(经典信号不支持排队队列,但是自定义信号可以),最大处理1个

驱动开发软件需要支持排队,因为每个信号与功能绑定,功能需要排队处理。经典信号目的是为了杀死进程,处理一次即可,无需排队。

信号通过屏蔽字,从未决态切换为递达态,系统将未决的1设置回0。系统会对正在处理的信号的屏蔽字临时设为1,等待信号处理完将屏蔽字设回0(避免如果捕捉函数中使用全局资源,信号多次执行出现异常)相同信号同时触发,最大可以处理两个,一个正在处理,一个被屏蔽,其他被丢弃。

如果将某个信号对应的屏蔽字位设置为1,该信号被阻塞,不允许递达。

用户可以自行设置信号屏蔽字,实现阻塞信号的效果。

屏蔽方式与忽略与捕捉有很大的不同。忽略与捕捉吸纳后已经递达了并处理完成了,但是屏蔽属于延迟处理,信号没有消失。

信号屏蔽设置:

    sigset_t set;    //信号集类型
    sigemptyset(&set);    //初始化0
    sigfillset(&set);    //初始化1
    sigaddset(&set,signo);    //将某个特定信号位置设置为1
    sigdelset(&set,signo);    //将某个信号位设置0
    int sigismember(&set,signo);    //返回特定信号位的位码
    sigprocmask(SIG_SETMASK,&newset,&oldset);    //设置替换进程的信号屏蔽字
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<signal.h>

int main(){
	sigset_t set,oset;
	sigemptyset(&set);
	sigaddset(&set,SIGQUIT);
	sigprocmask(SIG_SETMASK,&set,&oset);
	while(1)
		sleep(1);
	return 0;
}

四、高级信号与普通信号

高级信号:只要发出必然递达,无法屏蔽捕捉忽略

SIGKILL/9 必然杀死

SIGSTOP/19 必然挂起

硬件异常的信号SIGSEGV SIGFPE SIGBUS,用户发出触发可以屏蔽,但是系统触发不可屏蔽

查看当前进程实时的信号情况,应该查看未决信号集

    sigpending(&pset);    //调用此函数,系统会传出当前进程的未决信号集
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<signal.h>

int main(){
	sigset_t set,oset,pset;
	sigemptyset(&set);
	sigaddset(&set,SIGINT);//2
	sigaddset(&set,SIGQUIT);//3
	sigaddset(&set,SIGSEGV);//11
	sigaddset(&set,SIGALRM);//14
	sigprocmask(SIG_SETMASK,&set,&oset);
	while(1){
		sigpending(&pset);//传出进程未决
		for(int signo=1;signo<32;signo++)
			printf("%d",sigismember(&pset,signo));
		printf("\n");
		sleep(1);
	}
	return 0;
}

信号的处理不是实时的

系统发出的信号在内核层,而程序运行在用户层

1.进程执行于用户层,串行执行主函数代码

2.信号发送到内核层,等待处理

3.满足切换条件,切换到内核层

触发上下文切换(cpu权限切换)的三种事件:系统调用、软件中断(时间片完)、异常

4.完成调用、处置中断、处置异常

5.在返回用户层前,检查是否有未递达信号,如果有则处理

6.发现捕获函数在用户层使用,系统使用指令直接执行捕捉函数,避免切换(执行捕捉函数,系统并没有降低权限,避免了开销)

7.执行完毕通过SIG_RETURN指令返回内核层

8.执行上下文切换,返回用户空间

9.主函数从暂停的位置继续执行

一般情况下,主函数率先执行,但是执行过程中触发信号,系统调用捕捉函数,捕捉函数永远比主函数先执行完

主函数运行时使用当前进程的时间片,捕捉函数调用时主函数暂停使用当前进程的时间片(一个进程的时间片资源只有一份)

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

虚拟化数据恢复—误还原虚拟机快照怎么办?怎么恢复最新虚拟机数据?

虚拟化技术原理是将硬件虚拟化给不同的虚拟机使用&#xff0c;利用虚拟化技术可以在一台物理机上安装多台虚拟机。误操作或者物理机器出现故障都会导致虚拟机不可用&#xff0c;虚拟机中的数据丢失。 虚拟化数据恢复环境&#xff1a; 有一台虚拟机是由物理机迁移到ESXI上面的&a…

数据库管理-第188期 23ai:怎么用PGQL创建图(20240511)

数据库管理188期 2024-05-10 数据库管理-第188期 23ai:怎么用PGQL创建图&#xff08;20240511&#xff09;1 PGQL创建属性图1.1 PGQL属性图的元数据表1.2 创建一个PGQL属性图1.3 获取PGQL属性图的元数据 2 PGQL属性图3 官方示例演示3.1 插入数据3.2 创建PGQL属性图3.3 通过PGQL…

SpringBoot:SpringBoot原理

SpringBoot高级 SpringBoot配置 配置文件优先级 按照yaml>yml>properties的顺序加载 存在相同配置项,后加载的会覆盖先加载的 加载顺序越靠后,优先级越高 SpringBoot存在其他的多种方式进行配置,如下所示,越靠下优先级越高 1. Default properties (specified by s…

vm虚拟机安装网络适配器驱动卡死/无响应/无限等待状态

大部分原因都是以前的vm没有卸载干净所导致的&#xff0c;只需要使用CCleaner清楚干净就好 使用控制面板里的卸载把VM卸载干净 使用CCleaner软件删除干净注册表&#xff0c;这个软件百度很容易找到&#xff0c;只有十兆左右 打开下载好的软件&#xff0c;不需要注册码&#xff…

长安汽车:基于云器 Lakehouse 的车联网大数据平台建设

近年来随着智能汽车行业的迅速发展&#xff0c;数据也在呈爆炸式增长。长安大数据平台承接了长安在生产上大部分流量应用及日常生产业务应用。本文将分享长安汽车在车联网场景下大数据平台建设面临的一些挑战和具体落地的实践。 主要内容如下&#xff1a; 1. 背景介绍 2. 长…

Java数组:三种初始化

一.静态初始化 代码演示&#xff1a; //静态初始化:创建 赋值int[] a {1,2,3,4,5,6};System.out.println(a[0]); 二.动态初始化 代码演示&#xff1a; //动态初始化:包含默认初始化int[] b new int[10];b[0] 10;System.out.println(b[0]); //10System.out.println(b[1])…

25计算机考研院校数据分析 | 中南大学

中南大学&#xff08;Central South University&#xff09;&#xff0c;位于湖南省长沙市&#xff0c;是中华人民共和国教育部直属的全国重点大学 &#xff0c;中央直管副部级建制&#xff0c;位列国家“双一流”、“985工程”、“211工程”&#xff0c;入选国家“2011计划”牵…

MySQL前缀索引、脏页和干净页、COUNT(*)讨论、表删除内存问题

文章目录 如何加索引如何给身份证号添加索引 SQL语句变慢脏页 (Dirty Pages)干净页 (Clean Pages)为何区分脏页和干净页处理脏页管理策略 flush如何控制 为什么删除表数据后表文件大小不变问题背景核心原因数据存储方式参数影响 解决方案1. 调整innodb_file_per_table设置2. 使…

vs2019 cpp20 规范的线程头文件 <thread> 注释并探讨两个问题

&#xff08;1&#xff09;学习线程&#xff0c;与学习其它容器一样&#xff0c;要多读 STL 库的源码。很多知识就显然而然的明白了。也不用死记硬背一些结论。上面上传了一份注释了一下的 源码。主要是补充泛型推导与函数调用链。基于注释后的源码探讨几个知识点。 STL 库的多…

【SpringBoot】 什么是springboot(三)?springboot使用ajax、springboot使用reids

文章目录 SpringBoot第五章第六章1、springboot使用ajax2、springboot使用reids1、单机版**使用步骤**1-5步67-9步RedisTemplate使用RedisTemplate2、集群版开启集群项目配置1234-5第七章1、springboot文件上传使用步骤1-234-52、springboot邮件发送步骤1-23453、springboot拦截…

【智能算法】最优捕食算法(OFA)原理及实现

目录 1.背景2.算法原理2.1算法思想2.2算法过程 3.结果展示4.参考文献5.代码获取 1.背景 2017年&#xff0c;GY Zhu受到动物行为生态学理论启发&#xff0c;提出了最优捕食算法&#xff08;Optimal Foraging Algorithm, OFA&#xff09;。 2.算法原理 2.1算法思想 OFA灵感来源…

网络编程学习笔记1

文章目录 一、socket1、创建socket2、网络通信流程3、accept()函数4、signal()函数5、recv()函数6、connect()函数 二、I/O多路复用1.select模型2.poll模型3.epoll模型 注 一、socket 1、创建socket int socket(int domain,int type,int protocol); //返回值&#xff1a;一个…

微信小程序的Vant Weapp组件库(WeUI组件库)

一、定义&#xff1a; 是一套开源的微信小程序UI组件库。提供了一整套UI基础组件和业务组件&#xff0c;能够快速地搭配出一套风格统一的页面 二、使用&#xff1a; &#xff08;1&#xff09;&#xff08;找到.eslintrc.js 右键&#xff0c;在内件终端打开&#xff09;打开命…

|Python新手小白中级教程|第二十八章:面向对象编程(类定义语法私有属性类的继承与多态)(4)

文章目录 前言一、类定义语法二、私有方法和私有属性1.私有属性2.私有方法 三、类“继承”1.初识继承2.使用super函数调用父类中构造的东西 四、类“多态”1.多态基础2.子类不同形态3.使用isinstance函数与多态结合判断类型 总结 前言 大家好&#xff0c;我是BoBo仔吖&#xf…

RocketMQ学习笔记(一)

一、基本概念 生产者&#xff08;Producer&#xff09;&#xff1a;也称为消息发布者&#xff0c;是RocketMQ中用来构建并传输消息到服务端的运行实体&#xff0c;举例&#xff1a;发信者主题&#xff08;Topic&#xff09;&#xff1a;Topic是RocketMQ中消息传输和存储的顶层…

【全开源】Java知识付费教育付费资源付费平台公众号小程序源码

特色功能&#xff1a; 多样化的内容呈现&#xff1a;资源付费平台小程序支持图文、音视频、直播等多种形式的内容呈现&#xff0c;为用户提供了丰富的学习体验。直播课程&#xff1a;专家或讲师可以通过小程序进行在线授课&#xff0c;与用户实时互动&#xff0c;增强了学习的…

再有人说数字孪生大屏没有用,用这8条怼回去。

数字孪生大屏之所以受到欢迎&#xff0c;主要有以下几个原因&#xff1a; 实时数据可视化 数字孪生大屏可以将实时数据以直观的可视化形式展示出来&#xff0c;让用户能够一目了然地了解数据的状态和趋势。这样可以帮助用户更好地理解和分析数据&#xff0c;及时做出决策和调…

动态规划算法练习——计数问题

题目描述 给定两个整数 a 和 b&#xff0c;求 a 和 b 之间的所有数字中 0∼9 的出现次数。 例如&#xff0c;a1024&#xff0c;b1032&#xff0c;则 a 和 b 之间共有 9 个数如下&#xff1a; 1024 1025 1026 1027 1028 1029 1030 1031 1032 其中 0 出现 10 次&#xff0c;1 出现…

蓝桥杯-网络安全比赛(7)基础知识 HTTP、TTL、IP数据包、MSS、MTU、ARP、LLMNR、MDNS、NBNS。

1. IP中TTL值能够给我提供什么信息&#xff1f;2. IP头部中标志、13位偏移、32位源IP地址、目标IP、IP数据包格式&#xff0c;有多少字节3. IP头部中的16位标识是什么&#xff1f;4. MSS 和MTU分别有多大&#xff1f;5. 怎么获取路由IP信息&#xff1f;PING、NSLOOKUP、TRACERT…

day6Qt作业

人脸识别系统 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <opencv2/opencv.hpp> #include <iostream> #include <math.h> #include<opencv2/face.hpp> #include <vector> #include <map> #include <QMessag…