Linux 第三十章

🐶博主主页:@ᰔᩚ. 一怀明月ꦿ 

❤️‍🔥专栏系列:线性代数,C初学者入门训练,题解C,C的使用文章,「初学」C++,linux

🔥座右铭:“不要等到什么都没有了,才下定决心去做”

🚀🚀🚀大家觉不错的话,就恳求大家点点关注,点点小爱心,指点指点🚀🚀🚀

目录

信号量

如何理解信号量

信号 

信号的概念

信号的产生

singal

信号种类

0-31是普通信号

通过系统调用实现信号产生

Kill系统调用

raise

abort

异常发送信号

软件条件产生异常

alarm


信号量

信号量的本质是一把“计数器”

多个执行流(进程)看到的同一份资源,其实就是公共资源(由os提供的),如果并发访问同一份资源,会导致数据不一致的问题,我们就应该对公共资源进行保护,可以利用可同步和互斥(在多线程的时候着重理解)来解决。

互斥:任何时刻只允许一个执行流(进程)访问公共资源,加锁实现的

同步:多个执行流(进程)执行的时候,按照一定顺序执行,条件变量实现

被保护起来的公共资源,临界资源

访问临界资源的代码,临界区

非临界资源

非临界区

维护临界资源,其实就是在维护临界区

原子性:只有2两种状态,要么做,要么不做

如何理解信号量

信号量:表示资源数目的计数器,每一个执行流(进程)想要访问公共资源内的某一份资源,不应该让执行流(进程)直接访问,而是先申请信号量资源,其实就是先对信号量计数器进行--操作。本质上--就完成了,对资源的预定机制

如果申请不成功?我们的执行流(进程)被阻塞,等到申请成功

申请信号量资源是--

释放信号量资源是++

二元信号量:起到了互斥的效果(因为信号量资源只有一份),互斥锁

1.意味着,每一个进程都先看到同一个信号量资源
2.信号量本质也是公共资源
3.单个信号量
struct sem
{Int count;Task_strct* wait_queue
}

信号 

信号的概念

生活中的信号

红绿灯、下课铃声、狼烟、闹钟等等

我(进程)为什么知道信号呢?肯定我(进程)之前就了解过信号

信号没有产生的时候,其实我(进程)已经能够知道,怎么处理这个信号

信号的到来,我(进程)并清楚具体什么时候,信号到来相对于我(进程)正在做的工作,是异步产生的

信号产生了,我(进程)不一定立即处理它,而是我(进程)在合理的时候处理

我(进程)必须要一种能力,将已经到来的信号进行暂时保存

什么叫做信号

信号是向目标进程发送通知消息的一种机制

异步

简单理解:在同一个时间下,一个进程做一个事情的时候,另一个进程也在做事情

信号的产生

进程在运行的时候
例:
前台运行:./myprocess
后台运行:./myprocess &

前台进程只能有一个,后台进程可以有多个

在 Linux 终端中,jobs 是一个命令用于显示当前终端会话中正在运行的作业(进程)。当你在终端上启动一个作业时,它将在前台或后台运行。使用 jobs 命令可以列出这些作业的状态和编号。
以下是一些与 jobs 命令相关的常用操作:
1. jobs:列出当前终端会话中正在运行的进程。每个进程都有一个编号,并且会显示进程状态(运行中、停止等)。
2. bg <job编号>:将一个停止的进程切换到后台运行。<job编号> 是通过 jobs 命令列出的进程编号。
3. fg <job编号>:将一个在后台运行的进程切换到前台运行。<job编号> 是通过 jobs 命令列出的进程编号。
4. Ctrl+Z:将当前正在前台运行的进程暂停,并将其放置到后台。可以使用 jobs 命令查看该进程的状态和编号。
5. Ctrl+C:终止当前正在前台运行的进程(不会终止shell)。

OS怎么知道键盘有数据输入呢?
通过硬件向CPU发送中断号,CPU就能够知道键盘有数据输入

信号本质其实就是用软件,来模拟中断的行为

键盘有两种数据输入:

1.普通数据

2.组合键的输入(^C、^Z等)

信号在合适的时候处理

1.默认行为(例如,生活中红灯时,需要我们等一等)

2.忽略行为

3.自定义的(例如,在等红灯的时候,唱歌)

singal

在 Linux 中,signal() 函数用于注册信号处理函数,

该函数原型如下:

#include <signal.h>
void (*signal(int signum, void (*handler)(int)))(int);
* signum 表示要捕获的信号编号,比如 SIGINT、SIGTERM 等。
* handler 是一个指向函数的指针,该函数负责处理接收到的信号。
signal() 函数的返回类型是一个函数指针,指向当前信号的处理函数。在调用 signal() 函数时,会将之前注册的信号处理函数(如果有的话)替换为新的处理函数,并返回之前注册的处理函数。

下面是一个简单的示例,演示如何使用 signal() 函数来捕获 SIGINT 信号并设置相应的处理函数:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>void sigint_handler(int signum) {printf("Caught SIGINT signal, exiting...\n");exit(1);
}int main() {// 注册 SIGINT 信号处理函数signal(SIGINT, sigint_handler);printf("Press Ctrl+C to send a SIGINT signal...\n");while (1) {// 进程持续执行}return 0;
}


在示例中,我们定义了一个名为 sigint_handler 的函数,用于处理接收到的 SIGINT 信号。然后,在 main() 函数中使用 signal() 函数注册了这个信号处理函数。当用户按下 Ctrl+C 发送 SIGINT 信号时,该处理函数将被调用。

需要注意的是,signal() 函数在一些情况下可能会有一些不确定的行为,因此推荐使用更加可靠和灵活的 sigaction() 函数来注册信号处理函数。

信号种类

1) SIGHUP       2) SIGINT       3) SIGQUIT      4) SIGILL       5) SIGTRAP6) SIGABRT      7) SIGBUS       8) SIGFPE       9) SIGKILL     10) SIGUSR1
11) SIGSEGV     12) SIGUSR2     13) SIGPIPE     14) SIGALRM     15) SIGTERM
16) SIGSTKFLT   17) SIGCHLD     18) SIGCONT     19) SIGSTOP     20) SIGTSTP
21) SIGTTIN     22) SIGTTOU     23) SIGURG      24) SIGXCPU     25) SIGXFSZ
26) SIGVTALRM   27) SIGPROF     28) SIGWINCH    29) SIGIO       30) SIGPWR
31) SIGSYS      34) SIGRTMIN    35) SIGRTMIN+1  36) SIGRTMIN+2  37) SIGRTMIN+3
38) SIGRTMIN+4  39) SIGRTMIN+5  40) SIGRTMIN+6  41) SIGRTMIN+7  42) SIGRTMIN+8
43) SIGRTMIN+9  44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9  56) SIGRTMAX-8  57) SIGRTMAX-7
58) SIGRTMAX-6  59) SIGRTMAX-5  60) SIGRTMAX-4  61) SIGRTMAX-3  62) SIGRTMAX-2
63) SIGRTMAX-1  64) SIGRTMAX 

信号既有名字,又有值,说明是用宏定义的

没有0号信号(用来标识进程正常退出)

信号一共62个信号,是从0-31和34-64

0-31是普通信号

* SIGHUP (1):挂起信号,通常用于重启进程。
* SIGINT (2):中断信号,通常由Ctrl+C发出,用于停止进程。
* SIGQUIT (3):退出信号,通常由Ctrl+\发出,用于强制终止进程。
* SIGILL (4):非法指令信号,通常由进程执行无效的机器指令而引起。
* SIGTRAP (5):陷阱信号,通常用于调试。
* SIGABRT (6):异常终止信号,通常由abort()函数发出,表示进程存在严重问题需要终止。
* SIGBUS (7):总线错误信号,通常由非法内存访问引起。
* SIGFPE (8):浮点异常信号,通常由除零或其他数学错误引起。
* SIGKILL (9):强制终止信号,无法被捕获、忽略或阻塞,用于强制杀死进程。
* SIGUSR1 (10):用户定义信号1。
* SIGSEGV (11):段错误信号,通常由非法内存访问引起。
* SIGUSR2 (12):用户定义信号2。
* SIGPIPE (13):管道破裂信号,通常发生在进程向已关闭的管道写入数据时。
* SIGALRM (14):定时器信号,通常由alarm()函数或setitimer()函数发出。
* SIGTERM (15):终止信号,通常用于请求进程正常停止。
* SIGSTKFLT (16):堆栈错误信号,通常发生在堆栈溢出时。
* SIGCHLD (17):子进程状态改变信号,通常用于父进程监控子进程的状态。
* SIGCONT (18):继续执行信号,通常用于恢复被停止的进程。
* SIGSTOP (19):停止信号,无法被捕获、忽略或阻塞,用于暂停进程。
* SIGTSTP (20):挂起信号,通常由Ctrl+Z发出,用于暂停进程。
* SIGTTIN (21):后台读取信号,通常发生在后台进程尝试从终端读取数据时。
* SIGTTOU (22):后台写入信号,通常发生在后台进程尝试向终端写入数据时。
* SIGURG (23):紧急数据可读信号,通常发生在带外数据到达时。
* SIGXCPU (24):CPU时间限制信号,通常发生在进程超过CPU时间限制时。
* SIGXFSZ (25):文件大小限制信号,通常发生在进程超过文件大小限制时。
* SIGVTALRM (26):虚拟定时器信号,通常由setitimer()函数发出。
* SIGPROF (27):CPU时间分配信号,通常由setitimer()函数发出。
* SIGWINCH (28):窗口大小改变信号,通常发生在终端窗口大小改变时。
* SIGIO (29):异步I/O信号,通常用于异步I/O操作。
* SIGPWR (30):电源故障信号,通常发生在电源故障时。
* SIGSYS (31):系统调用错误信号,通常由进程执行非法的系统调用而引起。

每一个进程都有一张自己的函数指针数组表,数组的下标就和信号编号相关的

对于普通信号来讲,进程收到信号之后,进程要表示自己是否收到了某种信号

怎么表示是否收到呢?

可以使用位图的方式:0000 0000
比特位的位置,决定信号编号
比特位的内容,决定是否收到信号
在进程控制块中,有一个字段就是位图
task_struct
{//信号位图uint32_t sigmap;
}

OS向目标进程发送信号,其实就是向目标进程pcb的位图写信号

注意:无论信号有多少种产生方式,永远只能让OS向目标进程发送(OS是进程的管理者)

每个进程与信号相关都两个字段

1.函数指针数组

2.信号位图

34-64是实时信号

ctrl+z:默认暂停进程

ctrl+\:默认是终止进程

signal系统调用可以自定义,信号的处理方法

man 7 signal可以查看信号的默认处理方法

注意,像9号等信号不能自定义信号的处理方法

通过系统调用实现信号产生

Kill系统调用

kill 系统调用的原型如下:

#include <sys/types.h>
#include <signal.h>
int kill(pid_t pid, int sig);
其中,pid 参数指定要发送信号的目标进程的进程ID,sig 参数指定要发送的信号编号。

以下是 kill 系统调用的常见用法:
终止进程:使用 SIGKILL 信号(编号为9)可以强制终止指定进程。
kill(pid, SIGKILL);

请注意,kill 系统调用只能发送信号给具有合法权限的目标进程。此外,kill 系统调用返回0表示成功,返回-1表示出错,并设置相应的错误码。

raise

raise 是一个函数,用于向当前进程发送信号。它的原型如下:

#include <signal.h>
int raise(int sig);
其中,sig 参数指定要发送的信号编号。使用 raise 函数,可以向当前进程发送指定的信号。例如,以下代码将向当前进程发送 SIGTERM 信号(请求进程优雅地终止):
raise(SIGTERM);
与 kill 系统调用不同,raise 函数只能向当前进程发送信号,不能向其他进程发送信号。此外,raise 函数返回0表示成功,返回非零值表示出错,通常是因为指定了无效的信号编号。

abort

abort 是一个函数,用于向当前进程发送 SIGABRT 信号并终止进程。它的原型如下:
#include <stdlib.h>
void abort(void);
abort 函数会向当前进程发送 SIGABRT 信号,该信号的默认处理方式是生成一个核心转储文件并终止进程。abort 函数通常被用于在程序发生无法恢复的错误时,终止程序并生成一个核心转储文件以供调试分析。例如,在以下代码中,如果 malloc 函数返回空指针,我们可以使用 abort 函数终止程序并生成一个核心转储文件:
#include <stdlib.h>
void *ptr = malloc(size);
if (ptr == NULL) {fprintf(stderr, "Error: failed to allocate memory.\n");abort();
}
请注意,与 exit 函数不同,abort 函数不会调用已注册的退出处理程序,也不会关闭流或执行任何其他清理操作。因此,abort 函数应该谨慎使用,仅在必要时才使用。

异常发送信号

int a=10;
a/=0;如果异常不做处理(杀掉该进程),os会一直调度该进程,然后又出异常,循环往复。

寄存器!=寄存器里内容(进程上下文)

进程中的程序异常,进程一定会退出吗?默认情况下会退出,如果用signal自定义信号处理方法的话,就不一定了

如果进程出异常,和我们的编译器还有关系吗?进程会被OS直接终止了

一般出异常,是没有办法去处理这个异常,一般出异常,我们会显示这个异常,让我们自己看到

软件条件产生异常

alarm

#include <unistd.h>
unsigned int alarm(unsigned int seconds);
调用alarm函数可以设定一个闹钟,也就是告诉内核在seconds秒之后给当前进程发SIGALRM信号, 该信号的默认处理动
作是终止当前进程,如果 seconds 参数为0,则之前设置的闹钟将被取消,并返回剩余的闹钟时间。int main()
{alarm(1);int cnt=0;while(true){cout<<"alarm: "<<cnt++<<endl;}return 0;
}

操作系统中的时间:

1.所有用户的行为,都是以进程的形式在OS中表现的

2.OS只要把进程调度好,就能完成所有的用户任务

3.CMOS,周期性,高频率的,向CPU发送时钟中断

OS本质就是一个死循环,操作系统的执行,是基于硬件中断的!!!

 🌸🌸🌸如果大家还有不懂或者建议都可以发在评论区,我们共同探讨,共同学习,共同进步。谢谢大家! 🌸🌸🌸  

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

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

相关文章

CentOS使用Docker搭建Nacos结合内网穿透实现无公网IP远程登录本地管理平台

文章目录 1. Docker 运行Nacos2. 本地访问Nacos3. Linux安装Cpolar4. 配置Nacos UI界面公网地址5. 远程访问 Nacos UI界面6. 固定Nacos UI界面公网地址7. 固定地址访问Nacos Nacos是阿里开放的一款中间件,也是一款服务注册中心&#xff0c;它主要提供三种功能&#xff1a;持久化…

数据库——SQL SERVER(先学删库跑路)

目录 一&#xff1a;什么是数据库 二&#xff1a;为什么需要数据库 三&#xff1a;数据库的安装 四&#xff1a;学前必备知识 1. 数据库原理 2. 数据库与编程语言 3. 数据库与数据结构的区别 4. 连接 5. 有了编程语言为啥还要数据库 6. 初学者学习数据库的三个方面 …

交友软件源码-源码+搭建+售后,上线即可运营聊天交友源码 专业语聊交友app开发+源码搭建-快速上线

交友小程序源码是一种可以帮助开发者快速搭建交友类小程序的代码模板。它通常包括用户注册、登录、个人信息编辑、匹配推荐、好友聊天等常见功能&#xff0c;以及与后台数据交互的接口。使用这种源码可以极大地缩短开发时间&#xff0c;同时也可以根据自己的需求进行二次开发和…

SpringCloud Config 分布式配置中心

SpringCloud Config 分布式配置中心 概述分布式系统面临的——配置问题ConfigServer的作用 Config服务端配置Config客户端配置 可以有一个非常轻量级的集中式管理来协调这些服务 概述 分布式系统面临的——配置问题 微服务意味着要将单体应用中的业务拆分成一个个字服务&…

携号转网查询接口-实时批量检测-手机运营商归属查询API

新版携号转网查询接口支持批量&#xff0c;具体对接示例参考之前文章&#xff1a; 手机号码携号转网检测API查询接口【2024最新版】_携号转网api-CSDN博客 ● 通过手机号精准实时查询该号码转网前及转网后所归属运营商信息 ● 可查询号码是否为虚拟运营商手机号 ● 精准实时…

第十二届蓝桥杯省赛真题 Java A 组【原卷】

文章目录 发现宝藏【考生须知】试题 A: 相乘试题 B: 直线试题 C : \mathrm{C}: C: 货物摆放试题 D: 路径试题 E: 回路计数试题 F : \mathrm{F}: F: 最少砝码试题 G: 左孩子右兄弟试题 H : \mathrm{H}: H: 异或数列试题 I \mathbf{I} I 双向排序试题 J : \mathrm{J}: J: 分…

QX---mini51单片机学习---(9)中断系统

目录 1什么是中断 2中断系统在单片机系统中的作用 3如何使用单片机的中断系统 4实践 1什么是中断 RST P0想输出高电平接上拉电阻 2中断系统在单片机系统中的作用 3如何使用单片机的中断系统 可位寻址&#xff1a;IE中的EA可以直接&#xff0c;EA1&#xff1b; 外部中断&…

e行64位V11.17.4 安卓全局虚拟定位APP

e行最新版11.17.4 支持全局虚拟位置 小米手机 百度地图 高德地图 实测成功 其他app自测 不一定支持所有app 下载&#xff1a;https://www.123pan.com/s/HAf9-tsyCh.html

uni-app安卓本地打包个推图标配置

如果什么都不配置&#xff0c;默认的就是个推小鲸鱼图标 默认效果 配置成功效果 个推图标配置 新建目录 drawable-hdpi、drawable-ldpi、drawable-mdpi、drawable-xhdpi、drawable-xxhdpi、drawable-xxxhdpi 目录中存放图标 每个目录中存放对应大小的图标&#xff0c;大图…

centos7中如何优雅的动态切换jdk版本?

在 CentOS 7 中动态切换 JDK 版本可以通过多种方法实现&#xff0c;其中最常见的方法是使用 alternatives 命令&#xff0c;这是 CentOS 和其他基于 Red Hat 的系统中用于管理多个软件版本的标准工具。下面我会详细介绍如何使用 alternatives 命令来切换 JDK 版本。 步骤 1: 安…

STM32_HAL_系统定时器(SysTick)_实现计时

1介绍 系统定时器&#xff08;SysTick&#xff09;是ARM Cortex-M处理器系列中的一个特殊定时器&#xff0c;它不属于STM32F1系列微控制器的外设&#xff0c;而是处理器内部的一个组件。SysTick定时器的作用是为操作系统或其他需要精确时钟计数和中断服务的应用提供基础的时间…

山姆·奥特曼接受All-in Podcast采访

前言 在“All-in Podcast”播客中&#xff0c;OpenAI的CEO山姆奥特曼广泛讨论了人工智能的多个关键议题。他涉及了推理计算、开源模型的发展、GPT-5语言模型的进展&#xff0c;并对AI监管、全民基本收入&#xff08;UBI&#xff09;政策、智能体如何改变应用交互&#xff0c;以…

基于SpringBoot的全国风景区WebGIS按省展示实践

目录 前言 一、全国风景区信息介绍 1、全国范围内数据分布 2、全国风景区分布 3、PostGIS空间关联查询 二、后台查询的设计与实现 1、Model和Mapper层 2、业务层和控制层设计 三、WebGIS可视化 1、省份范围可视化 2、省级风景区可视化展示 3、成果展示 总结 前…

未来互联网:Web3的技术革新之路

引言 随着技术的不断发展和社会的日益数字化&#xff0c;互联网作为信息交流和社交媒介的重要平台已经成为我们生活中不可或缺的一部分。然而&#xff0c;传统的互联网架构在数据安全、隐私保护和去中心化等方面存在着诸多挑战。为了解决这些问题&#xff0c;Web3技术应运而生…

Ubuntu22.04怎么安装cuda11.3

环境&#xff1a; WSL2 Ubuntu22.04 问题描述&#xff1a; Ubuntu22.04怎么安装cuda11.3 之前是11.5 解决方案&#xff1a; 在Ubuntu 22.04上安装CUDA 11.3需要一些步骤&#xff0c;因为CUDA 11.3不是为Ubuntu 22.04官方支持的版本。但是&#xff0c;您仍然可以通过以下步…

python内置函数exec()和eval()区别

在Python中&#xff0c;eval() 和 exec() 都是内置函数&#xff0c;用于执行存储在字符串或对象中的Python代码&#xff0c;但它们之间也有一些区别。 eval() 语法&#xff1a;eval(expression, globalsNone, localsNone) expression&#xff1a;需要求值的字符串表达式。可…

面试题:调整数字顺序,使奇数位于偶数前面

题目&#xff1a; 输入一个整数数组&#xff0c;实现一个函数&#xff0c;来调整该数组中数字的顺序 使得所有奇数位于数组的前半部分&#xff0c;所有偶数位于数组的后半部分 算法1&#xff1a; 利用快速排序的一次划分思想&#xff0c;从2端往中间遍历 时间复杂度&#x…

【class2】人工智能初步(自然语言处理)

要实现从评价中提取高频关键词&#xff0c;并判别其正负面性&#xff0c;其实是通过人工智能领域中的一个分支&#xff1a;自然语言处理。 在了解自然语言处理之前&#xff0c;我们先来说说&#xff0c;什么是自然语言&#xff08;Natural Language&#xff09;&#xff1f;自…

快速入门:利用Go语言下载Amazon商品信息的步骤详解

概述 在这篇文章中&#xff0c;我们将深入探讨如何利用Go语言这一强大的工具&#xff0c;结合代理IP技术和多线程技术&#xff0c;实现高效下载Amazon的商品信息。首先&#xff0c;让我们来看看为什么选择Go语言作为开发网络爬虫的首选语言。 Go语言在网络开发中的特点 简洁…

springboot学习整理

视频&#xff1a;基础篇-01_springboot概述_哔哩哔哩_bilibili 介绍 spring boot 是spring提供的一个子项目&#xff0c;用于快速构建spring应用程序 spring构建&#xff1a; 1 导入依赖繁琐 &#xff1b; 2 项目配置繁琐 spring Framework: 核心 spring Boot :快速构建spring…