Linux —— 信号初识

news2024/12/27 11:28:36

Linux —— 信号初识

  • 什么是信号
  • 测试几个信号
  • signal函数
      • 函数原型
      • 参数说明
      • 返回值
      • 注意事项
      • 示例
  • 后台程序
  • 前台转后台
  • 检测输入
    • 中断向量表

我们今天来继续学习Linux的内容,今天我们要了解的是Linux操作系统中的信号:

什么是信号

信号是操作系统内核与进程之间进行异步通信的一种机制,它允许系统或进程向另一个进程发送简短的控制信息,以通知进程有特定的事件发生或要求进程采取某种行动信号是软中断,意味着它们是由软件生成的,并非直接由硬件触发。以下是Linux信号的一些关键概念:

  1. 信号的来源
  • 内核: 内核可以因各种事件自动发送信号给进程,如进程试图执行非法指令、访问无效内存地址、用户按下Ctrl+C终止进程等。
  • 进程: 进程可以通过系统调用kill()向自己或其他进程发送信号。
  • 终端: 用户在终端上执行操作,如按下Ctrl+C或Ctrl+Z,也会导致内核向前台进程发送信号SIGINT(中断)或SIGTSTP(停止)。
  • 硬件: 尽管信号主要由软件生成,某些硬件异常(如断电)也可以间接触发信号。
  1. 信号的目的

    • 通知事件: 信号用来告知进程某些状态变化或事件的发生,如子进程结束、定时器到期等。
    • 控制进程行为: 信号可以请求进程采取特定动作,比如终止、暂停、继续执行或调整优先级等。
  2. 信号的处理方式

    • 默认动作: 每个信号都有一个默认的行为,如SIGINT通常会导致进程终止。
    • 忽略: 进程可以选择忽略某些信号,即不对信号做出反应。
    • 自定义处理: 进程可以定义自己的信号处理函数,通过signal()sigaction()系统调用来指定信号的处理方式。
  3. 信号掩码和阻塞

    • 进程可以设置信号掩码来暂时阻止(阻塞)某些信号的传递,直到进程解除阻塞。
  4. 常见信号:

    • SIGINT (2): 当用户按下Ctrl+C时发送,通常用于中断进程。
    • SIGTERM (15): 用来请求进程正常终止。
    • SIGHUP (1): 挂起信号,通常在终端挂断时发送给与之相连的进程。
    • SIGKILL (9): 不能被忽略或阻塞,用于强制结束进程。
    • SIGSTOP (19): 停止进程,不能被捕获或忽略。

我们可以用kill -l来查看所有的信号:
在这里插入图片描述这里我们来看几个比较重要的:

  1. SIGHUP (1) - 挂起信号
    当终端线路挂断时发送给控制终端所属的进程组。通常用于通知进程配置文件可能已更改,需要重新加载。守护进程经常捕获此信号以实现优雅重启。
  2. SIGINT (2) - 中断信号
    当用户按下Ctrl+C时产生,请求进程中断当前操作并退出。默认情况下会导致进程终止。
  3. SIGQUIT (3) - 退出信号
    类似于SIGINT,但通常伴随着生成核心转储(core dump),用于调试。在终端下,通常是Ctrl+\ 发送此信号。
  4. SIGKILL (9) - 强制终止信号
    不能被捕获、忽略或阻塞,用于立即结束进程。当其他手段无法终止进程时使用。
  5. SIGTERM (15) - 终止信号
    一种温和的请求进程终止的信号,进程可以注册处理函数来自定义清理操作。是结束进程的首选方式。
  6. SIGSEGV (11) - 段错误信号
    当进程尝试访问不允许其访问的内存段时发送,通常指示程序中的内存访问错误。
  7. SIGALRM (14) - 闹钟信号
    与定时器相关联,当设定的定时器超时时发送。常用于实现定时任务或超时检测。
  8. SIGCHLD (17) - 子进程状态改变信号
    父进程接收到此信号,表明其子进程已经终止或停止。用于监控子进程状态并回收资源。
  9. SIGSTOP (19) - 停止信号
    强制进程停止执行。不能被忽略或被捕获,类似于暂停键,常用于调试。
  10. SIGCONT (18) - 继续执行信号
    使被SIGSTOP停止的进程恢复执行。通常配合SIGSTOP使用,用于控制进程的暂停与继续。
  11. SIGUSR1 和 SIGUSR2 (10, 31) - 用户自定义信号
    这两个信号留给用户自定义用途,可以用于进程间通信或触发特定的处理逻辑。

这些信号在系统编程中扮演着关键角色,理解它们有助于编写更稳定、可维护的代码,尤其是在需要处理进程间通信、异常情况或实现特定行为的场景中。

测试几个信号

我们创建一个cc文件:

#include<iostream>
#include<unistd.h>
using namespace std;

int main()
{
    while(true)
    {
        cout << "running process ..." << endl;
        sleep(1);
    }
}

我们用g++编译,运行一下:
在这里插入图片描述
我们**Ctrl+C**可以终止进程:
在这里插入图片描述
同时,我们重新运行,另开一个窗口:
在这里插入图片描述
我们也可以使用信号3:
在这里插入图片描述或者Ctrl + \
在这里插入图片描述

signal函数

为了验证Ctrl + C信号2是否是同一件事情,我们可以利用signal来验证:
在这里插入图片描述
在Linux中,signal()函数是一个用于处理信号的关键函数,它允许进程对操作系统发送的各种信号做出响应。信号是Linux和其他类UNIX系统中一种进程间通信(IPC)的方式,用于通知进程发生了某种事件,如用户请求终止进程、硬件故障、定时器到期等。下面是对signal()函数的基本介绍和使用方法:

函数原型

#include <signal.h>

typedef void (*sighandler_t)(int);

sighandler_t signal(int signum, sighandler_t handler);

参数说明

  • signum:要处理的信号编号,比如SIGINTSIGTERM等。
  • handler:当指定的信号发生时,系统调用的处理函数。它可以是以下几种:
  • SIG_DFL(默认处理):恢复信号的默认行为,如终止进程。
  • SIG_IGN(忽略信号):忽略此信号。
  • 自定义函数:一个用户自定义的函数指针,该函数原型通常为void function(int signum),其中signum是接收到的信号编号。

返回值

  • 如果成功,signal()返回之前为该信号设置的处理函数的地址。如果之前没有设置处理函数(即使用默认处理),则返回SIG_DFL;如果之前忽略了该信号,则返回SIG_IGN
  • 在某些系统上,如果提供了无效的signumhandler不是SIG_DFLSIG_IGN或有效的函数指针,signal()可能会失败并返回SIG_ERR

注意事项

  • signal()的行为在不同版本的POSIX标准和不同的Unix系统之间有所不同,特别是关于信号处理函数的重新安装性。某些系统遵循传统BSD语义,而另一些则遵循POSIX.1-1990或POSIX.1-2001语义。
  • 对于实时信号(如SIGRTMINSIGRTMAX之间的信号),推荐使用sigaction()函数来代替signal(),因为它提供了更精细的控制和更一致的跨平台行为。

示例

下面是一个简单的示例,展示了如何使用signal()来捕获SIGINT(通常是Ctrl+C)信号,并忽略它,使得进程在接收到此信号时终止:

#include<iostream>
#include<unistd.h>
#include <signal.h>
#include<cstdio>
using namespace std;

void signal_hander(int signum)
{
    printf("Caught SIGINT, but ignoring...\n");
    exit(0);
}

int main()
{
    signal(2,signal_hander);

    while(true)
    {
        cout << "process running ..." << endl;
        sleep(1);
    }
}

我们重新编译一下,运行一下:
在这里插入图片描述
我们又用信号2来试验一下:
在这里插入图片描述

后台程序

大家发现没有,如果我们运行process,此时我们输入命令行是没有用的:
在这里插入图片描述
如果我们不想这样,我们可以把它放在后台,只要后面带一个&就行:
在这里插入图片描述
此时再用Ctrl + C是无法结束后台进程的:
在这里插入图片描述此时,只用两种办法,第一种用信号,或者pkill + 进程名
在这里插入图片描述第二种,让后台程序回到前台:
在这里插入图片描述后台程序一运行时,会有一个编号,此时fg + 编号可以让后台程序回到前台:
在这里插入图片描述回到前台,就可以使用Ctrl + C。

前台转后台

快捷键Ctrl + Z可以让前台程序停止,转向后台:
在这里插入图片描述
命令行jobs可以看到所有的后台程序:
在这里插入图片描述如果我们想开启,我们使用bg + 序号
在这里插入图片描述就可以让程序在后台运行。

检测输入

我们输入Ctrl + Z等这些组合键,操作系统识别键盘输入的过程大致如下:

  1. 硬件层面:现代键盘通常通过USB或无线连接与计算机通信,以前的老式键盘可能使用PS/2接口。当用户按下键盘上的一个键时,键盘硬件会生成一个电信号,这个信号代表了特定按键的扫描码。
  2. 中断请求:键盘控制器将这个信号转换成键盘中断请求,并发送给计算机的中断控制器。中断是CPU对外部事件的一种快速响应机制,它能够暂停当前正在执行的任务,转而处理紧急的外部事件。
  3. 中断处理:CPU接收到中断请求后,会保存当前任务的状态(如程序计数器等),然后跳转到中断处理程序的入口地址执行。对于键盘中断,这个处理程序通常是操作系统的一部分。
  4. 读取扫描码:在中断处理程序中,操作系统会读取键盘控制器中的数据寄存器,获取按键的扫描码。扫描码是一个独一无二的标识,对应于键盘上的每一个键。
  5. 转换为ASCII码或虚拟键码:操作系统接着会将扫描码转换为操作系统内部可以理解的形式,如ASCII码(用于文本字符)或虚拟键码(用于功能键和特殊键)。这个过程可能涉及查表或其他映射机制。
  6. 事件队列与应用程序:转换后的字符或按键信息会被封装成一个事件,并放入系统的消息队列中。等待处理的事件包括按键按下和释放等。当应用程序(如文本编辑器)调用相应的API(如Windows的 GetMessage 或 Linux 的 select/poll)检查消息队列时,操作系统会将这些事件传递给应用程序。
  7. 应用程序响应:应用程序根据接收到的键盘事件执行相应的操作,比如在文本框中显示字符或响应快捷键命令。
  8. 释放中断:一旦中断处理完成,操作系统会恢复之前被中断的任务状态,继续执行。

整个过程确保了用户在键盘上的输入能够迅速、准确地被操作系统捕捉并传递给正在运行的应用程序。

中断向量表

这里面还有一个中间向量表:

中断向量表是计算机系统中一个非常关键的数据结构,它存储了所有中断服务程序(ISR,Interrupt Service Routines)的入口地址。这些中断服务程序负责处理各种硬件或软件触发的中断事件。以下是中断向量表的一些关键特性与作用:

  1. 内存中的固定位置:中断向量表通常位于内存中的一个固定位置,使得CPU在任何时候都能迅速访问到它。在某些体系结构中,比如x86,中断向量表可能位于低地址区域,便于快速响应中断。
  2. 条目结构:表中的每个条目对应一个中断类型或中断源,条目内包含的是相应中断服务程序的起始地址(或者是一个跳转指令,间接指向实际的中断处理程序)。每个条目可能占用2个、4个或更多字节,具体取决于处理器架构。
  3. 中断类型号与向量地址:中断类型号是一个标识特定中断的数字。在一些系统中,中断类型号乘以某个固定值(如4)可以得到该中断向量在中断向量表中的地址,这样CPU就能根据中断类型快速定位到正确的中断处理程序。
  4. 中断响应过程:当CPU检测到一个中断请求时,它会立即停止当前的任务执行,保存现场(即当前的处理器状态),然后根据中断类型号查询中断向量表,获取中断服务程序的入口地址,并跳转到该地址开始执行中断处理程序。
  5. 复位与初始化:在系统启动或复位时,中断向量表的基地址会被初始化到一个预定义的位置。某些处理器(如ARM Cortex-M系列)允许通过VTOR(向量表偏移寄存器)来重新定位中断向量表的位置,以适应不同的系统配置需求。
  6. 固定与动态分配:在一些简单的系统中,中断向量表可能是静态定义的,而在更复杂的系统中,部分中断向量可能支持动态分配,允许操作系统或固件在运行时安装或更改中断服务例程的地址。

中断向量表的设计和管理是确保系统能够高效、可靠地响应各种内外部事件的基础,对于维持系统的实时性和稳定性至关重要。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

45. UE5 RPG 增加角色受击反馈

在前面的文章中&#xff0c;我们实现了对敌人的属性的初始化&#xff0c;现在敌人也拥有的自己的属性值&#xff0c;技能击中敌人后&#xff0c;也能够实现血量的减少。 现在还需要的就是在技能击中敌人后&#xff0c;需要敌人进行一些击中反馈&#xff0c;比如敌人被技能击中后…

Mac基于Docker-ubuntu构建c/c++编译环境

编译环境安装和使用被充分验证&#xff0c;如有期望补充的内容欢迎留言评论。 目录 前言 Docker desktop下载安装 修改镜像源 选择ubuntu镜像 docker容器启动 参数说明: 宿主机与docker容器文件共享 宿主机与docker容器拷贝文件 为 Ubuntu 配置 ssh、vim、make 相关工…

基于Springboot的校园生活服务平台(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的校园生活服务平台&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构…

Maven 在项目的 pom.xml 文件中 指定 阿里云的景象仓库

配置 在 项目的 pom.xml 文件中添加如下配置即可 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation&…

计数排序,基数排序,桶排序

目录 计数排序: 基数排序&#xff1a; 桶排序: 计数排序: 计数排序是一种非比较型整数排序算法&#xff0c;特别适用于一定范围内的整数排序。它的核心思想是使用一个额外的数组&#xff08;称为计数数组&#xff09;来计算每个值的出现次数&#xff0c;然后根据这些计数信…

Nginx 从入门到实践(2)——Rewrite重写

Nginx Rewrite Rewrite重写 Nginx Rewriteurl组成说明Rewrite基本概述Rewrite使⽤场景rewrite优点 Rewrite配置语法location匹配概述 if指令if 判断指令语法nginx以及if 判断可使用的全局变量 set命令return指令 url组成说明 https://cn.bing.com/search?qNginxRewrite&P…

VMware虚拟机忘记密码重置--centos7x

centos7虚拟机重置root密码 操作流程 操作流程 重启虚拟机&#xff0c;在如下页面键入e键&#xff1a; 2. 在如下位置添加&#xff1a;init/bin/sh&#xff0c;添加完成后键入Ctrlx启动 依次键入下列命令&#xff1a; mount -o remount,rw / #重新挂载/目录 passwd root #修改…

【微服务】分布式事务(通过Seata解决分布式事务问题)

分布式事务 分布式事务Seata微服务集成SeataXA模式XA模式使用 AT模式AT模式实现 分布式事务 在分布式系统中&#xff0c;如果一个业务需要多个服务合作完成&#xff0c;而且每一个服务都有事务&#xff0c;多个事务必须同时成功或失败&#xff0c;这样的事务就是分布式事务&am…

选择深度学习框架:TensorFlow 2 vs PyTorch

TensorFlow 2 vs PyTorch 选择深度学习框架&#xff1a;TensorFlow 2 vs PyTorchTensorFlow 2概述TensorFlow 2的优点TensorFlow 2的缺点 PyTorch概述PyTorch的优点PyTorch的缺点 选择建议对于选择困难症的人&#xff0c;我给你们的答案——PyTorch选择理由&#xff1a;结论&am…

Mac 电脑安装 Raptor 流程图软件的方法

0. 安装逻辑 &#xff08;1&#xff09;运行 raptor&#xff0c;本质上需要 mac 能够运行 windows 程序&#xff0c;因此需要安装 .NET Runtime 7.0&#xff0c;这是微软程序运行必须的文件。 &#xff08;2&#xff09;运行 raptor 还需要安装依赖文件 mono-libgdiplus。 &am…

python从0开始学习(三)

目录 前言 1、类型转换 1.1 隐式类型转换 1.2 显式类型转换 2、eval函数 总结 前言 上篇我们讲了python中的变量与常量&#xff0c;以及变量类型。本篇文章将接着往下讲。 1、类型转换 python中的数据类型转换包括两种&#xff1a;隐式类型转换和显式类型转换。 1.1 隐式…

微信IDE vscode插件:获取插件位置,并打开文件

背景 有没有觉得在微信开发工具里面添加一些插件可以很方便。因为微信IDE的编辑本身是依赖vscode开发&#xff0c;所以编写vscode插件自然可以在微信IDE使用。这样做好处就是可以满足到自己一些开发使用习惯。 1.获取插件的目录位置 那么如何获取插件里面的目录&#xff0c;…

分布式领域计算模型及SparkRay实现对比

目录 一、分布式计算领域概览 二、Spark计算模型分析 三、Ray计算模型分析 3.1 需求分析 3.2 系统设计 3.3 系统实现 四、总结 一、分布式计算领域概览 当前分布式计算模型主要分为以下4种&#xff1a; Bulk Synchronous Parallel Model&#xff08;块同步并行模型&…

常用语音识别开源四大工具:Kaldi,PaddleSpeech,WeNet,EspNet

无论是基于成本效益还是社区支持&#xff0c;我都坚决认为开源才是推动一切应用的动力源泉。下面推荐语音识别开源工具&#xff1a;Kaldi&#xff0c;Paddle&#xff0c;WeNet&#xff0c;EspNet。 1、最成熟的Kaldi 一个广受欢迎的开源语音识别工具&#xff0c;由Daniel Pove…

开发体育赛事直播平台,研发技术选型与架构设计实现方案

本文将深入探讨“东莞梦幻网络科技”现成体育直播源码的技术实现方案&#xff0c;如何为用户提供流畅、互动、个性化的观赛体验。 一、技术栈选择&#xff1a;强强联合的基石1、后端开发&#xff1a;采用Java与PHP作为主要开发语言。Java以其强大的企业级应用支持&#xff0c;保…

C++证道之路第十七章输入输出和文件

一、C输入和输出概述 C 提供了丰富的输入/输出&#xff08;I/O&#xff09;功能&#xff0c;这些功能主要通过 <iostream> 头文件中的类和对象来实现。 1.流和缓冲区 C把程序输入和输出看作字节流。输入时&#xff0c;程序从输入流中抽取字节&#xff1b;输出时&#…

使用 FFmpeg 从音视频中提取音频

有时候我们需要从视频文件中提取音频&#xff0c;并保存为一个单独的音频文件&#xff0c;我们可以借助 FFmpeg 来完成这个工作。 一、提取音频&#xff0c;保存为 mp3 文件: 要使用 FFmpeg 从音视频文件中提取音频&#xff0c;并将 ACC 编码的音频转换为 MP3 格式&#xff0…

【数据结构(邓俊辉)学习笔记】列表02——无序列表

文章目录 0.概述1.插入与构造1.1 插入1.1.1 前插入1.1.2后插入1.1.3 复杂度 1.2 基于复制构造1.2.1 copyNodes()1.2.2 基于复制构造1.2.3 复杂度 2.删除与析构2.1 删除2.1.1 实现2.1.2 复杂度 2.2 析构2.2.1 释放资源及清除节点2.2.2 复杂度 3.查找3.1 实现3.2 复杂度 4.唯一化…

FFmpeg学习记录(四)——SDL音视频渲染实战

1.SDL使用的基本步骤 SDL Init/sDL _Quit()SDL_CreateWindow()/SDL_DestoryWindow()SDL CreateRender() SDL_Windows *windows NULL;SDL_Init(SDL_INIT_VIDEO);window SDL_CreateWindow("SDL2 Windows",200,200, 640,480,SDL_WINDOW_SHOWN);if(!window) {printf(&…

【C语言回顾】数据在内存中的存储

前言1. 概述2. 大小端字节序和字节序判断2.1 大端字节序&#xff08;Big-Endian&#xff09;2.2 小端字节序&#xff08;Little-Endian&#xff09;2.3 判断字节序的示例 3. 数据在内存中的存储3.1 整数在内存中的存储3.2 浮点数在内存中的存储 结语 ↓ 上期回顾: 【C语言回顾】…