处理 Linux 信号:进程控制与异常管理的核心
个人主页:chian-ocean
文章专栏-Linux
前言:
在 Linux 操作系统中,信号是用于进程间通信的一种机制,能够向进程发送通知,指示某些事件的发生。信号通常由操作系统内核、硬件中断或其他进程发送。接收和处理信号是 Linux 系统中进程控制和资源管理的一个重要组成部分。

信号的保存
信号递送(Delivery)
- 信号的处理动作称为信号递送(Delivery)。
- 这意味着在 Linux 系统中,当信号发生时,它会被传递到目标进程并执行相应的操作。递送是信号处理的第一步,是信号机制中至关重要的一部分。
2. 信号未决(Pending)
-
信号从产生到递送之间的状态,称为信号未决(Pending)。
- 当信号发送到一个进程,但该进程因某些原因(如信号被屏蔽或进程正在执行其他操作)暂时无法处理信号时,这个信号就会处于未决状态,等待后续处理。
-
Linux内核是通过一个一个位图编标记信号,进而储存信号

block位图(通常是一个sigset_t类型的数据结构)用于记录当前进程被阻塞的信号。每一位代表一个特定的信号,当该位被设置为1时,表示该信号处于被阻塞状态;如果该位是0,则表示该信号未被阻塞,可以递送给进程。pending 位图(Pending Bitmap)用于记录进程中待处理的信号。位图是一个位数组,其中每个比特代表一个信号的状态(是否待处理)。headler代表的是该信号所需要用的方法。
sigset_t

定义:
sigset_t 是一个适用于信号管理的基本数据类型,通常在头文件 <signal.h> 中定义。它的具体实现依赖于系统,但通常是一个位图或位集合,每一位表示一个信号的状态(是否被屏蔽、是否待处理等)。
sigset_t 的用途
- 信号屏蔽:用于设置进程的信号掩码(signal mask),即哪些信号被屏蔽,不允许递送。例如,通过
sigprocmask函数修改信号掩码。 - 信号检查:通过
sigpending()函数检查进程是否有待处理的信号。 - 信号操作:例如,
sigaddset()和sigdelset()等函数可以用来向信号集添加或移除特定信号
常用函数与 sigset_t 的操作
在 Linux 系统中,sigset_t 是一个数据类型,用于表示信号集,通常用于管理信号的掩码。它用于存储一个进程中多个信号的集合,可以用来表示进程阻塞的信号、待处理的信号等。
sigset_t 类型定义
sigset_t 是一个适用于信号管理的基本数据类型,通常在头文件 <signal.h> 中定义。它的具体实现依赖于系统,但通常是一个位图或位集合,每一位表示一个信号的状态(是否被屏蔽、是否待处理等)。
sigset_t 的用途
sigset_t 主要用于以下几个方面:
- 信号屏蔽:用于设置进程的信号掩码(signal mask),即哪些信号被屏蔽,不允许递送。例如,通过
sigprocmask函数修改信号掩码。 - 信号检查:通过
sigpending()函数检查进程是否有待处理的信号。 - 信号操作:例如,
sigaddset()和sigdelset()等函数可以用来向信号集添加或移除特定信号。
sigset_t 的操作
以下是一些与 sigset_t 相关的常见操作函数:
-
sigemptyset(sigset_t ,*set): 初始化一个空的信号集(即不包含任何信号)。sigset_t set; sigemptyset(&set); // 将 set 设为空信号集 -
sigfillset(sigset_t ,*set): 初始化一个包含所有信号的信号集。sigset_t set; sigfillset(&set); // 将 set 设为包含所有信号的信号集 -
sigaddset(sigset_t ,*set, int signo): 将指定的信号添加到信号集。sigset_t set; sigemptyset(&set); sigaddset(&set, SIGINT); // 将 SIGINT 添加到 set 中 -
sigdelset(sigset_t ,*set, int signo): 从信号集移除指定的信号。sigdelset(&set, SIGINT); // 从 set 中删除 SIGINT -
sigismember(const sigset_t ,*set, int signo): 检查指定的信号是否在信号集中。if (sigismember(&set, SIGINT)) {// 如果 SIGINT 在 set 集合中 }
控制sigset_t常见函数
sigprocmask(int how, const sigset_t ,*set, sigset_t ,*oldset): 修改进程的信号掩码(即屏蔽哪些信号)。
-
sigismember(const sigset_t ,*set, int signo): 检查指定的信号是否在信号集中。if (sigismember(&set, SIGINT)) {// 如果 SIGINT 在 set 集合中 }-
how:参数控制信号掩码的设置方式:
SIG_BLOCK:把 set中的信号添加到blocked中(blocked= blocked | set)。SIG_UNBLOCK:从 blocked中删除 set中的信号(blocked= blocked & set)。SIG_SETMASK:block= set
sigset_t set, oldset; sigemptyset(&set); sigaddset(&set, SIGINT); sigprocmask(SIG_BLOCK, &set, &oldset); // 阻塞 SIGINT -
-
sigpending(sigset_t g*set): 获取当前进程的未决信号集,返回一个信号集,表示所有尚未被处理的信号。sigset_t set; sigpending(&set); // 获取当前进程的待处理信号
屏蔽字的实例
#include <iostream>
#include <signal.h>
#include <unistd.h>// 函数:用于打印信号集中的信号状态
void pendingprint(sigset_t &pending)
{// 遍历信号集中的每个信号,从31到1(Linux支持的信号范围通常是1到31)for(int i = 31; i >= 1; i--){// 检查信号集pending中是否包含第i个信号if (sigismember(&pending, i)){std::cout << "1"; // 如果信号存在,打印"1"}else{std::cout << "0"; // 如果信号不存在,打印"0"}}std::cout << "\n"; // 换行打印结果
}int main()
{// 定义信号集,用于保存信号掩码、旧掩码和待处理信号集sigset_t mask, old_mask;sigset_t pending;// 初始化信号集mask为空信号集sigemptyset(&mask);// 将SIGINT信号(Ctrl+C触发的中断信号)添加到信号集mask中sigaddset(&mask, SIGINT);// 将SIGINT信号添加到当前进程的信号掩码中,即阻塞SIGINT信号// 并保存原来的信号掩码到old_mask中sigprocmask(SIG_BLOCK, &mask, &old_mask);int cnt = 0;while(true){// 获取当前进程的待处理信号集,并保存在pending中sigpending(&pending);// 调用pendingprint函数,打印待处理信号集中的信号pendingprint(pending);// 每次睡眠1秒钟sleep(1);// 计数器,每次增加1if(++cnt == 5){// 在第20次循环时,解除阻塞SIGINT信号std::cout << "SIGINT UNLOCK" << std::endl;// 恢复之前的信号掩码,即解除对SIGINT信号的阻塞sigprocmask(SIG_SETMASK, &old_mask, NULL);}}std::cout << "QUIT..." << std::endl;return 0;
}
- 代码中我会每秒打印
penging中的数据 - 第三秒中的时候,像进程发送2好信号,
penging中会显示为处理的信号,在bolck中该信号被阻塞,也就是被屏蔽。 - 代码执行第五秒中的时候,进程阻塞信号解除,由于
penging中有未处理的信号,就会执行2好号信号。

信号的捕捉
信号捕捉的时机
- 进程是否正在执行系统调用。
- 进程是否在空闲状态(如调用
sleep())。 - 信号是否被发送。
- 信号是否被阻塞或保留直到适当时机。
- 信号是否导致进程退出或崩溃时处理。
信号执行的流程

- 用户模式:进程在用户模式下执行,直到某个信号因中断、异常或系统调用被触发。
- 内核模式:当信号被触发时,操作系统切换到内核模式,进行信号的处理。内核会完成信号的处理并准备将进程返回到用户模式之前的状态。
- 信号处理:在信号处理过程中,内核会调用信号处理函数(如
do_signal())。如果信号需要对用户指定的特定处理,系统会在信号处理时调用相应的函数。 - 处理返回:信号处理函数执行完毕后,操作系统会通过
sigreturn系统调用返回到用户模式。 - 恢复执行:操作系统恢复用户模式下的程序执行,从中断的地方继续执行。
用户态和内核态的切换
用户态到内核态:
- 触发方式:通过系统调用(如
read(),write())或硬件中断(如键盘中断)。 - 过程:
- 当用户程序调用系统调用时,会执行一个软中断(例如x86的
int 0x80指令)。 - 内核会保存当前用户程序的上下文(如程序计数器、堆栈指针等),并设置内核模式。
- 进入内核代码执行,执行完后,准备返回用户态。
- 当用户程序调用系统调用时,会执行一个软中断(例如x86的
内核态到用户态:
- 触发方式:内核任务完成后,需要返回用户程序继续执行。
- 过程:
- 内核将处理结果返回用户进程,并恢复用户进程的上下文。
- 恢复用户态的堆栈指针、程序计数器等寄存器,并将程序从内核模式切换回用户模式。
- 继续执行用户进程。
sigaction
sigaction是处理粒度更加强大的一个系统调用。

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
参数:
signum:指定信号的编号(例如,SIGINT、SIGTERM等)。act:指向struct sigaction结构体的指针,用于指定新的信号处理程序及其设置。oldact:指向struct sigaction结构体的指针,用于保存之前的信号处理程序(可选)。如果不关心旧的处理程序,可以传递NULL。
返回值:
- 成功时,返回
0。 - 失败时,返回
-1并设置errno。
struct sigaction {void (*sa_handler)(int); // 信号处理函数void (*sa_sigaction)(int, siginfo_t *, void *); // 备用信号处理函数sigset_t sa_mask; // 用于指定在信号处理期间要阻塞的信号int sa_flags; // 信号处理行为的标志void (*sa_restorer)(void); // 不常用,保留字段
};
代码示例:
#include<iostream>
#include<signal.h>
#include<unistd.h>
#include<cstring>
#include<cstdlib>// 信号处理函数,当捕获到信号时执行
void headler(int signo)
{// 输出捕获到的信号类型std::cout << "catch signal --- signo :" << signo << std::endl;sleep(4);// 退出程序,返回 1 作为退出状态exit(1);
}int main()
{ // 定义 sigaction 结构体,用于设置新的信号处理行为 (sa) 和保存旧的信号处理行为 (osa)struct sigaction sa, osa;// 使用 memset 将结构体清零,确保结构体中的字段没有未初始化的值memset(&sa, 0, sizeof(sa));memset(&osa, 0, sizeof(osa));// 清空信号集,准备设置信号屏蔽sigemptyset(&sa.sa_mask);// 将 SIGINT 信号添加到屏蔽信号集中,意思是处理 SIGINT 时,其他信号会被阻塞sigaddset(&sa.sa_mask, SIGINT);// 设置自定义的信号处理函数,当 SIGINT 信号触发时,调用 headler 函数sa.sa_handler = headler;// 注册信号处理函数,绑定 SIGINT 信号与信号处理函数 headler,同时保存原来的信号处理方式sigaction(SIGINT, &sa, &osa);// 进入一个无限循环,模拟进程运行while (true){// 输出当前进程的 pid,表示进程正在运行std::cout << "Process Running ...: pid: " << getpid() << std::endl;// 每 1 秒输出一次,模拟进程持续运行sleep(1);}return 0;
}
信号处理函数 headler:
- 该函数定义了如何处理接收到的
SIGINT信号。当捕获到SIGINT信号时,会输出信号编号,然后模拟处理过程(休眠 4秒),最后退出程序并返回状态码 1。
信号处理结构体 sigaction:
sigaction是用来定义和控制信号处理方式的结构体。通过它可以设置信号的处理函数、信号掩码等信息。- 通过
memset清空结构体,确保没有未初始化的字段。 - 使用
sigemptyset和sigaddset配置信号屏蔽集,指定在处理SIGINT时,阻塞其他信号(在本例中,仅阻塞SIGINT自身)。
sigaction 调用:
sigaction(SIGINT, &sa, &osa)设置SIGINT信号的处理程序为headler,并保存原来的信号处理方式(虽然在这里我们没有使用osa来恢复原处理方式)。
进程运行:
- 进入无限循环
while(true),打印当前进程的pid(进程 ID),模拟一个长时间运行的进程。 - 每秒输出一次进程信息,并通过
sleep(1)使程序每秒钟暂停一次。
信号捕获时期的相关问题
- 信号递归:信号处理期间默认屏蔽当前信号,避免递归调用。
void headler(int signo)
{int cnt = 20;while(cnt --){std::cout << "catch signal --- signo :" << signo <<std:: endl;sleep(1);}exit(1);
}
- 在这里面
headler方法进行循环,并且休眠一秒,在此期间持续像进发送2号信号。

- 同时可以及逆行进行多信号屏蔽。
sigaddset(&sa.sa_mask,1);
sigaddset(&sa.sa_mask,3);
sigaddset(&sa.sa_mask,4);

- 信号丢失:如果信号未及时处理,可能丢失,尤其是长时间阻塞时。
- 系统调用中断:信号可能中断系统调用,导致
EINTR错误。 - 信号屏蔽问题:如果未正确设置信号掩码,可能导致信号被过多屏蔽或错误处理。
信号pending表的处理时机
- 信号在调用处理方法之前会被制空
#include<iostream>
#include<signal.h>
#include<unistd.h>
#include<cstring>
#include<cstdlib>// 声明一个全局的 sigset_t 变量,用于保存挂起的信号
sigset_t pending;// 打印挂起信号的状态(1表示挂起,0表示没有挂起)
void PrintPending()
{sigset_t set;sigpending(&set); // 获取当前挂起的信号集合// 遍历信号编号,打印每个信号的状态for (int signo = 31; signo >= 1; signo--){if (sigismember(&set, signo)) // 检查该信号是否在挂起集合中std::cout << "1"; // 如果挂起,打印 1elsestd::cout << "0"; // 如果没有挂起,打印 0}std::cout << "\n"; // 输出换行
}// 信号处理程序
void headler(int signo)
{int cnt = 5; // 设置循环次数为 5while (cnt--) // 每次处理信号时,循环 5 次{PrintPending(); // 打印当前挂起的信号状态sleep(1); // 休眠 1 秒,模拟信号处理的过程std::cout << "catch signal --- signo :" << signo << std::endl; // 输出捕获到的信号编号}exit(1); // 信号处理完毕后退出程序
}int main()
{signal(SIGINT, headler); // 设置 SIGINT 信号的处理函数为 headlersigset_t mask, old_mask;sigemptyset(&mask); // 初始化信号集 mask,清空所有信号sigaddset(&mask, SIGINT); // 将 SIGINT 信号加入到 mask 中,表示屏蔽 SIGINT 信号sigprocmask(SIG_BLOCK, &mask, &old_mask); // 阻塞 SIGINT 信号,并保存原信号掩码到 old_maskint cnt = 1;while (true){sigpending(&pending); // 获取当前挂起的信号PrintPending(); // 打印挂起信号的状态sleep(1); // 程序每秒输出一次状态cnt++; // 计数器加 1std::cout << "Process Running ...: pid: " << getpid() << std::endl; // 输出当前进程的 PIDif (cnt % 5 == 0) // 每经过 5 次循环{PrintPending(); // 再次打印挂起信号的状态sleep(1); // 休眠 1 秒std::cout << "SIGINT UNLOCK" << std::endl; // 输出解锁 SIGINT 信号的提示sigprocmask(SIG_SETMASK, &old_mask, NULL); // 恢复原来的信号掩码,解除对 SIGINT 的阻塞}}std::cout << "QUIT..." << std::endl; // 输出退出提示return 0;
}
- 程序启动后,会阻塞 SIGINT 信号。
- 每秒钟,程序打印当前进程的 PID 和挂起的信号状态。
- 每经过 5 次循环,解除对 SIGINT 信号的阻塞,允许信号被捕获并处理。
- 捕获到 SIGINT 信号后,
headler会打印挂起信号状态,处理信号并退出。

相关文章:
处理 Linux 信号:进程控制与异常管理的核心
个人主页:chian-ocean 文章专栏-Linux 前言: 在 Linux 操作系统中,信号是用于进程间通信的一种机制,能够向进程发送通知,指示某些事件的发生。信号通常由操作系统内核、硬件中断或其他进程发送。接收和处理信号是 Li…...
【蓝桥杯每日一题】4.1
🏝️专栏: 【蓝桥杯备篇】 🌅主页: f狐o狸x "今日秃头刷题,明日荣耀加冕!" 今天我们来练习二分算法 不熟悉二分算法的朋友可以看:【C语言刷怪篇】二分法_编程解决算术问题-CSDN博客 …...
分享系列项目的基础项目
本人分享了一系列的框架项目,它们共同需要依赖这个公共基础,结构如下图所示: 其中: audit: JPA的审计信息基础类auth:认证授权相关类config: 包括redis配置,client中token配置,openai文档配置…...
为 MinIO AIStor 引入模型上下文协议(MCP)服务器
Anthropic 最近宣布的模型上下文协议 (MCP) 将改变我们与技术交互的方式。它允许自然语言通信替换许多任务的复杂命令行语法。不仅如此,语言模型还可以总结传统工具的丰富输出,并以人类可读的形式呈现关键信息。MinIO 是世界领先的…...
spring boot前后端开发上传文件时报413(Request Entity Too Large)错误的可能原因及解决方案
可能原因及解决方案 1. Spring Boot默认文件大小限制 原因:Spring Boot默认单文件最大为1MB,总请求体限制为10MB。解决方案: 在application.properties中配置:spring.servlet.multipart.max-file-size10MB # 单文件最大 spring…...
数据结构实验1.1: 顺序表的操作及其应用
这里写自定义目录标题 一、实验目的二、注意事项三、实验内容(一)问题描述(二)基本要求 四,操作步骤(一)使用visual studio集成环境编写程序 五,示例代码六,运行效果 一、…...
基于yolov11的汽车损伤检测系统python源码+onnx模型+评估指标曲线+精美GUI界面
【算法介绍】 基于YOLOv11的汽车损伤检测系统是一种先进的计算机视觉技术,旨在快速准确地识别汽车的各种损伤类型。该系统利用YOLOv11模型的强大性能,实现了对车辆损伤的精确检测与分类。 该系统能够识别的损伤类型包括裂纹(crackÿ…...
OpenCV 图形API(2)为什么需要图形API?
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 G-API背后的动机 G-API模块为OpenCV带来了基于图的执行模型。本章简要描述了这种新模型如何在两个方面帮助软件开发者:优化和移植图像处理算法…...
基于Spring Boot的平面设计课程在线学习平台系统的设计与实现(LW+源码+讲解)
专注于大学生项目实战开发,讲解,毕业答疑辅导,欢迎高校老师/同行前辈交流合作✌。 技术范围:SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容:…...
esp32 idf中的外部组件
通常外部组件都定义在“main/idf_component.yml”里面。 例如小智智能语音的外部组件: ## IDF Component Manager Manifest File dependencies:waveshare/esp_lcd_sh8601: "1.0.2"espressif/esp_lcd_ili9341: "1.2.0"espressif/esp_lcd_gc9a01…...
【JavaEE】MyBatis - Plus
目录 一、快速使用二、CRUD简单使用三、常见注解3.1 TableName3.2 TableFiled3.3 TableId 四、条件构造器4.1 QueryWrapper4.2 UpdateWrapper4.3 LambdaQueryWrapper4.4 LambdaUpdateWrapper 五、自定义SQL 一、快速使用 MyBatis Plus官方文档:MyBatis Plus官方文档…...
经典卷积神经网络LeNet实现(pytorch版)
LeNet卷积神经网络 一、理论部分1.1 核心理论1.2 LeNet-5 网络结构1.3 关键细节1.4 后期改进1.6 意义与局限性二、代码实现2.1 导包2.1 数据加载和处理2.3 网络构建2.4 训练和测试函数2.4.1 训练函数2.4.2 测试函数2.5 训练和保存模型2.6 模型加载和预测一、理论部分 LeNet是一…...
【2】数据结构的单链表章
目录标题 单链表的定义单链表的初始化单链表的建立头插法创建尾插法创建 查找操作按序号查找按内容查找 插入操作删除操作合并操作 单链表总代码与调试 单链表的定义 结点(Node)的定义:数据域(data)和指针域ÿ…...
Linux(十一)fork实例练习、文件操作示例及相关面试题目分享
一、fork实例练习 1、思考下面这段代码的打印结果是什么? #include<stdio.h> #include<unistd.h> #include<assert.h> #include<stdlib.h>int main(){int i0;for(;i<2;i){fork();printf("A\n");} exit(0); }所以一共打印6…...
android Fragment使用
在 Android Fragment 中,导入 id(findViewById)并给控件赋值的逻辑通常应该写在 onViewCreated() 方法中,而不是 onCreateView()。 Fragment 生命周期 & 适合的位置 方法作用适合的操作onCreateView()创建并返回 Fragment 的…...
open3d教程 (三)点云的显示
官方文档位置: Visualization - Open3D 0.19.0 documentationhttps://www.open3d.org/docs/release/tutorial/visualization/visualization.html核心方法: o3d.visualization.draw_geometries([几何对象列表]) import open3d as o3dprint("Load …...
根据模板将 Excel 明细数据生成 Txt 文档|邮件合并
在日常办公中,我们常常会遇到需要批量生成文档的任务。以往,若要将 Excel 中的每一条数据都转化为单独的文档,且文档部分内容依据 Excel 数据动态变化,手动操作不仅繁琐,还容易出错。现在,有一种便捷的方法…...
【学Rust写CAD】22 双圆径向渐变的结构体(two_circle_radial_gradient.rs)
源码 //two_circle_radial_gradient.rs //! 定义双圆径向渐变的结构体和相关功能/// 表示一个双圆径向渐变的源 /// /// 该结构体描述了两个圆之间的渐变,支持矩阵变换和颜色查找表优化 #[derive(Debug, Clone, PartialEq)] pub struct TwoCircleRadialGradientSou…...
LVGL Dropdown和Calendar详解
LVGL Dropdown和Calendar详解 一、Dropdown详解创建和初始化设置下拉框选项获取选项获取选中项文本:获取选中项索引:设置选中项: 事件处理其他功能和样式设置设置下拉按钮样式:设置下拉框方向:设置最大高度:…...
AISEO (GEO )中的知识图谱
一、知识图谱在AI SEO中的概念与结构 1. 知识图谱是什么? 定义:知识图谱(Knowledge Graph)是一种以图结构组织的语义网络,由实体(Entity)、**关系(Relation)和属性&…...
Vulnhub-zico2靶机打靶记录
本篇文章旨在为网络安全渗透测试靶机教学。通过阅读本文,读者将能够对渗透Vulnhub系列zico2靶机有一定的了解 一、信息收集阶段 靶机下载地址:https://download.vulnhub.com/zico/zico2.ova 因为靶机为本地部署虚拟机网段,查看dhcp地址池设…...
(041)05-01-自考数据结构(20331)树与二叉树大题总结
实际考试中,计算题约占40%,推理题约占30%,算法设计题约占30%。建议重点练习遍历序列相关的递归分治解法, 知识拓扑 知识点介绍 一、计算题类型与解法 1. 结点数量计算 题型示例: 已知一棵完全二叉树的第6层有8个叶子结点,求该二叉树最多有多少个结点? 解法步骤: 完…...
Python----机器学习(KNN:使用数学方法实现KNN)
一、原理 以下是K最近邻(K-Nearest Neighbors,简称KNN)算法的基本流程,用于对给定点进行分类预测。 1. 获得要预测的点 point_predict 。 2. 计算训练点集 point_set_train 中各点到要预测的点 表 l ist_L2_distance 。 3. 对 poi…...
网络攻防快速入门笔记pwn | 02 栈溢出题型 | 2.2 ret2libc
上一篇:网络攻防快速入门笔记pwn | 02 栈溢出题型 | 2.1 ret2text和ret2shellcode 下一篇:网络攻防快速入门笔记pwn | 02 栈溢出题型 | 2.3 ret2syscall 欢迎关注~ ret2libc 一、 什么是ret2libc(一)ret2lib的概念(…...
Edge浏览器快速开启IE模式
一些老旧的网站,仅支持Internet Explorer(IE)浏览器访问。 然而,出于安全性的考虑,可能会遇到限制IE浏览器使用的情况。 Microsoft Edge浏览器提供了兼容性配置,可以通过IE模式访问这些网站。 以下是两种…...
技术长期主义:用本分思维重构JavaScript逆向知识体系(一)Babel、AST、ES6+、ES5、浏览器环境、Node.js环境的关系和处理流程
基础不牢,地动山摇,逆向越久,越发现基础的重要性,本系列,回顾js逆向基础,让自己的知识体系更加系统化。 以下是 Babel、AST、ES6、ES5、浏览器环境、Node.js环境 的关系和流程的详细说明及图表:…...
Oracle 数据库系统全面详解
Oracle 数据库是全球领先的关系型数据库管理系统(RDBMS),由 Oracle 公司开发。它为企业级应用提供了高性能、高可用性、安全性和可扩展性的数据管理解决方案。 目录 一、Oracle 数据库体系结构 1. 物理存储结构 主要组件: 存储层次: 2. …...
LeetCode 解题思路 29(Hot 100)
解题思路: 映射关系建立:创建一个哈希表存储数字到字母的映射。递归参数: 给定字符串 digits、结果集 result、当前路径 path、当前位置 start。递归过程: 当当前位置 start 等于 digits 长度时,说明已经遍历完 digi…...
使用Python解析PPT文件并生成JSON结构详解
引言 PowerPoint(PPT)文件的自动化处理是办公自动化和数据提取的常见需求。本文将介绍如何通过Python的python-pptx库,将PPT文件的样式、结构、文本内容等信息解析为标准化的JSON格式,为后续的自动化处理、数据迁移或样式复用提供…...
LabVIEW永磁同步电机性能测试系统
开发了一种基于LabVIEW的永磁同步电机(PMSM)性能测试系统的设计及应用。该系统针对新能源汽车使用的电机进行稳态性能测试,解决了传统测试方法成本高、效率低的问题,实现了测试自动化,提高了数据的准确性和客观性。 …...
