【探索Linux】—— 强大的命令行工具 P.16(进程信号 —— 信号产生 | 信号发送 | 核心转储)

在这里插入图片描述

阅读导航

  • 引言
  • 一、概念
    • (1)基本概念
    • (2)kill -l命令(察看系统定义的信号列表)
  • 二、产生信号
    • (1)通过终端按键产生信号
      • -- 信号产生
      • -- Core Dump(核心转储)
    • (2)调用系统函数向进程发信号
      • kill( ) 函数
      • abort( ) 函数
    • (3) 由软件条件产生信号
      • alarm( ) 函数
    • (4)硬件异常产生信号
  • 温馨提示

引言

在现代社会中,信号无处不在。我们的生活充满了各种各样的信号,它们指引着我们前进的方向,使我们能够了解周围环境的变化。正如在计算机编程中一样,Linux进程信号也是一种重要的信号,它们扮演着相似的角色。

想象一下,在繁忙的城市街道上行驶,交通信号灯是我们最熟悉的信号之一。当红灯亮起时,我们知道需要停下来等待;而绿灯的出现则意味着可以继续前行。这些信号通过明确的方式向司机传达信息,确保道路上的交通有序进行。

类似地,Linux进程信号也是一种用于进程间通信和控制的手段。它们是操作系统通过发送特定信号给进程来通知其发生了某种事件或请求进行某种操作的机制。这些信号可以用于中断进程、终止进程、重新启动进程以及执行其他与进程相关的操作

Linux进程信号的产生和发送是一个复杂而精密的过程,它涉及操作系统内部的多个组件和机制。深入理解信号的工作原理对于编写高效、稳定的程序至关重要。通过掌握信号的概念和使用方法,我们可以更好地利用操作系统提供的功能,实现各种任务的灵活管理和交互。

在本文中,我们将探讨Linux进程信号的基本概念、信号的产生方式以及如何通过编程发送信号给进程。通过深入了解信号的工作原理,我们将能够更好地理解操作系统的内部机制,并在编写程序时更加灵活地利用信号来实现我们的目标。让我们一起踏上这个关于Linux进程信号的精彩探索之旅吧!

一、概念

(1)基本概念

Linux中的信号是一种软件中断。当操作系统发送一个信号给一个进程时,该进程会立即停止正在执行的任务,并跳转到一个特定的信号处理函数中进行处理。信号可以用于中断进程、终止进程、重新启动进程以及执行其他与进程相关的操作。

(2)kill -l命令(察看系统定义的信号列表)

在这里插入图片描述
Linux中共有64个不同的信号,它们被分为两类:标准信号和实时信号。标准信号的编号从1到31,实时信号的编号从32到64。每个信号都有一个唯一的名称和一个对应的数字编号,例如SIGINT表示中断信号,其编号为2。

在 Linux 中,这些宏定义通常可以在 <signal.h> 头文件中找到。每个信号都有一个编号和一个宏定义名称。

对于信号的产生条件和默认处理动作的详细说明,可以通过查看 signal(7) 的手册页来获取。可以使用以下命令来查看:

man 7 signal

这将打开关于信号的手册页,其中包含了关于信号产生条件、默认处理动作以及如何使用 signal() 函数进行自定义信号处理的详细说明。
在这里插入图片描述
在Linux中,我们可以通过编写信号处理函数来对信号进行处理。信号处理函数是在接收到信号后自动调用的函数,用于处理信号并执行相应的操作。当一个信号产生时,操作系统通常会暂停当前进程的执行,保存进程的状态,然后跳转到信号处理函数中执行相应的操作。

二、产生信号

Linux中的信号可以由以下三种方式产生

  1. 用户按下终端键(如Ctrl+C),操作系统会将一个SIGINT信号发送给前台进程组中的所有进程;

  2. 进程调用kill()系统调用,向指定进程或进程组发送信号;

  3. 操作系统本身发现了某些异常情况,如进程访问非法内存地址、除零错误等,就会向进程发送相应的信号。

(1)通过终端按键产生信号

– 信号产生

在终端中,你可以通过按下特定的组合键来向正在运行的程序发送信号。其中最常用的是以下几个:

  1. Ctrl+C:产生 SIGINT 信号,通常用于中断当前程序的执行。
  2. Ctrl+Z:产生 SIGTSTP 信号,通常用于挂起当前程序的执行。
  3. Ctrl+\:产生 SIGQUIT 信号,通常用于请求当前程序退出,并生成 core 文件。

通过在终端中按下这些组合键,你可以模拟产生这些信号,从而触发相应的信号处理动作。

– Core Dump(核心转储)

Core Dump(核心转储)是指在程序异常终止时,将程序的内存状态转储到一个特殊文件中。这个文件称为 core 文件,它包含了程序在崩溃前的内存映像。

当程序发生严重错误、段错误(Segmentation Fault)或其他类似的问题导致程序崩溃时,操作系统会生成一个 core 文件。这个文件可以被用于调试程序,通过分析 core 文件可以了解程序崩溃时的内存状态,有助于定位和修复错误。

core 文件的生成受到操作系统的控制,通常在以下情况下会生成 core 文件:

  1. 程序显式地调用 abort() 函数。
  2. 程序收到 SIGQUIT 或 SIGILL 等信号。
  3. 程序发生段错误(Segmentation Fault)。

默认情况下,core 文件的生成是被启用的。但是,有时可能已经被禁用或限制了大小。你可以使用以下命令来检查系统的 core 文件配置:

ulimit -a | grep core

在这里插入图片描述

如果输出中显示了 core file size,则表示 core 文件生成是启用的,并且会显示 core 文件的最大大小限制。

(2)调用系统函数向进程发信号

kill( ) 函数

要向进程发送信号,可以使用系统函数kill()kill()函数的原型如下:

#include <sys/types.h>
#include <signal.h>

int kill(pid_t pid, int sig);

其中,pid参数是目标进程的进程ID(PID),sig参数是要发送的信号编号。

以下是一个示例代码,演示如何使用kill()函数向进程发送信号:

#include <sys/types.h>
#include <signal.h>
#include <stdio.h>

int main() {
    pid_t pid = 1234; // 替换为目标进程的实际进程ID

    if (kill(pid, SIGINT) == 0) 
    {
        printf("成功发送信号给进程 %d\n", pid);
    } 
    else 
    {
        perror("发送信号失败");
    }

    return 0;
}

在上面的示例中,kill()函数用来向进程ID为pid的进程发送SIGINT信号(中断信号)。如果函数返回值为0,则表示成功发送信号。否则,可以使用perror()函数输出错误信息。

abort( ) 函数

abort()是一个C标准库函数,用于引发程序的异常终止。调用abort()函数会导致程序生成core文件(如果core文件生成被启用)并退出。abort()函数的原型如下:

#include <stdlib.h>

void abort(void);

abort()函数会向进程发送SIGABRT信号,这是一个特殊的终止信号,通常用于表示程序遇到了严重错误,并主动请求终止。

当调用abort()函数时,系统会进行一系列的处理操作,包括终止当前进程、生成core文件、关闭文件等。然后,程序将立即退出。

以下是一个示例代码,演示了如何使用abort()函数:

#include <stdio.h>
#include <stdlib.h>

int main() {
    printf("开始执行程序\n");

    // 模拟一个错误条件
    int divisor = 0;
    if (divisor == 0) {
        printf("除数为零,程序终止\n");
        abort();
    }

    // 正常执行的代码
    printf("正常执行的代码\n");

    return 0;
}

在上面的示例中,当除数为零时,程序调用了abort()函数导致程序异常终止。在这种情况下,将会输出"除数为零,程序终止",然后程序会生成core文件(如果core文件生成被启用)并退出。

(3) 由软件条件产生信号

在Linux中,由软件条件产生信号是通过使用信号处理函数来实现的。信号是一种软件中断,用于通知进程发生了某个事件。下面是一个简单的示例代码,展示了如何在Linux中由软件条件产生信号:

#include <stdio.h>
#include <unistd.h>
#include <signal.h>

void signal_handler(int signal_num) {
    printf("Received signal: %d\n", signal_num);
}

int main() {
    // 注册信号处理函数
    signal(SIGUSR1, signal_handler);

    printf("Waiting for signal...\n");
    sleep(10);  // 模拟程序执行的一段时间

    return 0;
}

在上面的示例中,首先使用signal函数注册了一个信号处理函数signal_handler,该函数会在接收到SIGUSR1信号时被调用。然后,程序进入休眠状态sleep(10),等待信号的到来。可以使用以下命令发送SIGUSR1信号给该程序:

$ kill -SIGUSR1 <pid>

其中,<pid>是运行该程序的进程ID。当程序接收到信号后,就会执行信号处理函数,并打印出接收到的信号编号。

alarm( ) 函数

在Linux中,alarm()函数可以用来设置一个定时器,当定时器到时后,会给进程发送一个SIGALRM信号。alarm()函数的原型如下:

unsigned int alarm(unsigned int seconds);

其中,seconds参数指定了定时器的时间,单位是秒。如果seconds为0,则会取消之前设置的定时器。

以下是一个简单的示例代码,展示了如何使用alarm()函数来实现定时器功能:

#include <stdio.h>
#include <unistd.h>
#include <signal.h>

void signal_handler(int signal_num) {
    printf("Received signal: %d\n", signal_num);
}

int main() {
    // 注册信号处理函数
    signal(SIGALRM, signal_handler);

    // 设置定时器,时间为5秒
    alarm(5);

    printf("Waiting for alarm...\n");
    pause();  // 等待信号的到来

    return 0;
}

在上面的代码中,首先注册了一个信号处理函数signal_handler,当接收到SIGALRM信号时就会执行该函数。然后使用alarm()函数设置定时器,时间为5秒。接着,程序进入休眠状态pause(),等待信号的到来。可以看到,在5秒后,程序会收到SIGALRM信号,并执行对应的信号处理函数。

需要注意的是,alarm()函数只能设置一个全局定时器,如果需要同时多个定时器,可以考虑使用setitimer()函数。此外,调用alarm()函数会取消之前设置的定时器,如果需要保留之前的定时器,可以使用setitimer()ITIMER_VIRTUALITIMER_REAL选项。

(4)硬件异常产生信号

在Linux中,硬件异常通常由操作系统内核检测到,并通过信号来通知相关进程。下面是一些常见的硬件异常和相应的信号:

  1. 除零异常(Divide-by-Zero):当执行除法操作时除数为零时触发的异常。操作系统会向进程发送SIGFPE信号,表示浮点异常。

  2. 非法指令异常(Illegal Instruction):当执行非法或无效的指令时触发的异常。操作系统会向进程发送SIGILL信号,表示非法指令。

  3. 段错误异常(Segmentation Fault):当进程访问了未分配给它的内存空间或者试图向只读内存写入数据时触发的异常。操作系统会向进程发送SIGSEGV信号,表示段错误。

  4. 总线错误异常(Bus Error):当进程试图访问非法的物理内存地址或者对不支持的对齐方式进行访问时触发的异常。操作系统会向进程发送SIGBUS信号,表示总线错误。

  5. 浮点异常(Floating-Point Exception):当执行浮点计算出现溢出、下溢或非法操作时触发的异常。操作系统会向进程发送SIGFPE信号,表示浮点异常。

这些硬件异常信号可以被进程捕获并进行相应的处理。通过设置信号处理函数,进程可以在收到硬件异常信号时采取适当的措施,如记录日志、恢复状态或退出程序等。

🚨注意硬件异常的处理通常是由操作系统内核负责的,应用程序通常无法直接控制硬件异常的触发和处理。应用程序可以通过注册信号处理函数来处理与硬件异常相关的信号,但具体的处理方式受限于操作系统和硬件平台的约束

温馨提示

感谢您对博主文章的关注与支持!如果您喜欢这篇文章,可以点赞、评论和分享给您的同学,这将对我提供巨大的鼓励和支持。另外,我计划在未来的更新中持续探讨与本文相关的内容。我会为您带来更多关于Linux以及C++编程技术问题的深入解析、应用案例和趣味玩法等。如果感兴趣的话可以关注博主的更新,不要错过任何精彩内容!

再次感谢您的支持和关注。我们期待与您建立更紧密的互动,共同探索Linux、C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!
在这里插入图片描述

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

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

相关文章

软件设计师:计算机组成与体系结构之计算机基础知识

计算机基础知识 数据的表示 码制及进制转换 原码&#xff1a;将数值转成二进制反码&#xff1a;正数与原码完全相同&#xff1b;负数&#xff0c;除了符号位其他位取反补码&#xff1a;正数与原码完全相同&#xff1b;负数&#xff0c;在补码的基础上加1移码&#xff1a;补码…

JMeter—HTTP压测

一、创建线程组 右击-->添加-->Threads(Users)-->线程组 下面对比较重要的几个参数&#xff0c;讲解下&#xff1a; 名称&#xff1a; 就是给你这个线程组起名字。 线程数&#xff1a;指压力测试时候模拟几个用户测试接口。 Ramp-Up&#xff1a;这里指几秒运行完上面的…

Linux:Ubuntu虚拟机安装详解:VMware下的逐步指南

目录 1. centOS系统 2. ubuntu系统 1. 下载Ubuntu映像 step1 step2 step3 2. 新建虚拟机 step1 step2 Step3 step4 step5 step6 内存 内核 映像 显示 网络 3. 网络配置 NAT模式 本机IP获取 ​编辑 bridge模式 4. 开启虚拟机 5. 虚拟机常用配置 语言 …

linux的netstat命令和ss命令

1. 网络状态 State状态LISTENING监听中&#xff0c;服务端需要打开一个socket进行监听&#xff0c;侦听来自远方TCP端口的连接请求ESTABLISHED已连接&#xff0c;代表一个打开的连接&#xff0c;双方可以进行或已经在数据交互了SYN_SENT客户端通过应用程序调用connect发送一个…

The Bridge:从临床数据到临床应用(预测模型总结)

The Bridge:从临床数据到临床应用&#xff08;预测模型总结&#xff09; 如果说把临床预测模型比作临床数据和临床应用之间的一座“桥梁”&#xff0c;那它应该包括这样几个环节&#xff1a;模型的构建和评价、模型的概率矫正、模型决策阈值的确定和模型的局部再评价。 模型的构…

大金仓数据库:kingbase学习

kingbase学习 1.简介2. 安装3. 基础使用3.1 客户端连接3.2 表数据测试3.2.1 建表创建字段备注 3.2.2 数据写入测试3.2.2 查询测试json查询测试 4.springboot实战4.1 maven依赖4.2 连接配置4.3 mybatis-plus测试4.4 liquibase整合(1). 使用pg方式替换kingbase驱动上面(2). 修改j…

路由器DHCP分配IP地址规则

路由器DHCP分配IP地址的机制&#xff1a; 先设置一个IP地址池&#xff0c;假设是192.168.1.100-192.168.1.199一共100个。 来一个请求&#xff0c;看一下是不是以前请求过的地址&#xff0c;如果是&#xff0c;还是返回以前给过的IP&#xff0c;然后将到期时间(有些路由器默认…

45岁后,3部位“越干净”,往往身体越健康,占一个也要恭喜!

众所周知&#xff0c;人的生命有长有短&#xff0c;而我们的身体健康状态&#xff0c;也同样会受到年龄的影响&#xff0c;就身体的年龄层次而言&#xff0c;往往需要我们用身体内部的干净程度来维持&#xff0c;换句话说就是&#xff1a;若是你的身体内部越干净&#xff0c;那…

LeetCode-1689. 十-二进制数的最少数目 C/C++实现 超详细思路及过程[M]

&#x1f388;归属专栏&#xff1a;深夜咖啡配算法 &#x1f697;个人主页&#xff1a;Jammingpro &#x1f41f;记录一句&#xff1a;上一篇博客这里好像没改&#xff0c;那就不改了。 文章目录 LeetCode-1689. 十-二进制数的最少数目&#x1f697;题目&#x1f686;题目描述&…

CPU+GPU多样化算力,ZStack Cloud助力游戏精酿核心业务上云

游戏精酿通过ZStack Cloud云平台提供高性能、高可用的云主机、云存储和云网络&#xff1b;前期通过超融合架构快速构建云基础设施&#xff0c;来支持Jira、Redis等关键业务&#xff1b;并实现对原有私有云平台业务的替代&#xff0c;按需将原有私有云业务滚动迁移到ZStack Clou…

每日一题(LeetCode)----链表--链表最大孪生和

每日一题(LeetCode)----链表–链表最大孪生和 1.题目&#xff08;2130. 链表最大孪生和&#xff09; 在一个大小为 n 且 n 为 偶数 的链表中&#xff0c;对于 0 < i < (n / 2) - 1 的 i &#xff0c;第 i 个节点&#xff08;下标从 0 开始&#xff09;的孪生节点为第 (n…

Linux 6.7全面改进x86 CPU微码加载方式

导读最近&#xff0c;社区在清理 Linux 上的 Intel/AMD x86 CPU 微代码加载方面做了大量的工作&#xff0c;这些工作现已合并到 Linux 6.7 中。 由于在启动时加载 CPU 微代码对于减少不断出现的新 CPU 安全漏洞以及有时解决功能问题非常重要&#xff0c;Thomas Gleixner 最近开…

如何将Postman API转换JMeter进行扩展

可扩展性 Postman测试无法扩展。如果您的集合中有很多请求&#xff0c;Postman / Newman将使用1个线程&#xff08;用户&#xff09;并按顺序执行这些请求&#xff0c;而不是使用多个线程并发执行。 性能测试能力 由于可扩展性限制&#xff0c;Postman不适合API性能测试。性…

TYPE-C、PD原理

一、Type-C简介以及历史 自1998年以来&#xff0c;USB发布至今&#xff0c;USB已经走过20个年头有余了。在这20年间&#xff0c;USB-IF组织发布N种接口状态&#xff0c;包括A口、B口、MINI-A、MINI-B、Micro-A、Micro-B等等接口形态&#xff0c;由于各家产品的喜好不同&#x…

【分布式】分布式中的时钟

一、物理时钟 vs 逻辑时钟 时钟的存在主要是为了标识事件的发生顺序。 分布式系统不使用物理时钟记录事件&#xff0c;分布式系统中每个节点记录的时间并不一样&#xff0c;即使设置了 NTP 时间同步节点间也存在毫秒级别的偏差 所以需要有另外的方法记录事件顺序关系&#x…

2024年天津天狮学院专升本护理学专业《护理学基础》考试大纲

天津天狮学院2024年护理学专业高职升本入学考试《护理学基础》考试大纲 一、考试性质 《护理学基础》专业课程考试是天津天狮学院护理专业高职升本入学考试的必考科目之一&#xff0c;其性质是考核学生是否达到了升入本科继续学习的要求而进行的选拔性考试。 《护理学基础》考…

现代 C++ 函数式编程指南

现代 C 函数式编程指南 什么是 柯里化 &#xff08;Curry&#xff09;什么是 部分应用 &#xff08;Partial Application&#xff09; 二元函数 &#xff08;Partial Application&#xff09;参数排序 &#xff08;Partial Application&#xff09; 应用场景 计算碳衰减周期求年…

Web前端 -----【Vue】(vue组件基础)一文带你了解组件的创建、注册、使用(包括组件的嵌套)

目录 前言 什么是组件 为什么使用组件化开发 组件的使用 组件的使用分为三个步骤 创建组件 为什么配置项中的data不能使用直接对象的形式&#xff0c;必须使用function&#xff08;重点&#xff01;&#xff01;&#xff01;面试喜欢问&#xff09; 注册组件 使用组件 …

【Element】el-switch开关 点击弹窗确认框时状态先改变----点击弹窗取消框失效

一、背景 需求&#xff1a;在列表中添加定期出账的开关按钮&#xff0c;点击开关时&#xff0c;原来的状态不改变&#xff0c;弹出弹窗&#xff1b;点击弹窗取消按钮&#xff1a;状态不改变&#xff0c;点击弹窗确定按钮&#xff1a;状态改变&#xff0c;并调取列表数据刷新页…

JavaWeb学习(未完结)

文章目录 一、基本概念1.1 动态Web网站简介1.2 web应用程序1.3 静态web1.4 动态web 二、web服务器2.1 技术2.2 应用服务器2.3 安装 jdk8 三、Tomcat3.1 安装 Tomcat93.2 文件说明3.3 启动并使用Tomcat3.4 关闭Tomcat3.5 可能遇到的问题3.6 配置3.6.1 修改测试访问的网页地址3.6…