admin管理员组

文章数量:1794759

【Linux】解析在【进程PCB】中是如何实现【信号的处理方式(抵达/未决/阻塞)】

前言 大家好吖,欢迎来到 YY 滴 系列 ,热烈欢迎! 本章主要内容面向接触过C++的老铁 主要内容含:

一.基础知识:信号的处理动作< 递达/未决/阻塞 >

  • 信号递达 (Delivery):实际 执行信号 的处理动作
  • 信号未决状态(Pending):信号从产生到递达 之间 的状态
  • 信号阻塞 (Block):进程可以选择阻塞 (Block )某个信号。

注意: 被阻塞 的信号产生时将 保持在未决状态 ,直到进程解除对此信号的阻塞,才执行递达的动作.

  • 注意,阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作

二.信号处理动作< 抵达/未决/阻塞 >在内核中的表示

1.示意图&作用机制介绍&信号集sigeset_t介绍

  • 每个进程PCB中 都有如下图所示三张表,分别叫做 阻塞信号集,未决信号集,处理动作集 ,对应各个信号(1-64)
  • block&pending&handler表 ,分别表示 阻塞(block) 未决(pending),还有一个函数指针表示处理动作
  • 信号集(sigeset_t): 这个类型可以表示每个信号的“有效”或“无效”状态; 非0即1
  1. 在阻塞信号集中“有效”和“无效”的含义是该信号是否被阻塞
  2. 而在未决信号集中“有效”和“无效”的含义是该信号是否处于未决状态

信号集表表的作用机制:

  1. 信号产生时,pending位图为1;递达后设置为0——只要block位图为1,就无法递答
  2. 阻塞信号,设置block位图为1;阻塞状态解除后设置成0;

2.演示< 抵达/未决/阻塞 >在三张表中的表示

  • 演示:

3.如何改变信号的默认实现动作

【1】实现原理:设置信号的【默认处理函数】变成【自定义函数】
  • 每个进程PCB中 都有如下图所示三张表,分别叫做 阻塞信号集,未决信号集,处理动作集 ,对应各个信号(1-64)
  • 其中handler表中存储的是函数指针,指向对应的处理动作
  • 原理:我们只要改变我们要改变的信号(例如信号2)handler表中的指针,由SIG_IGN指向的函数,改成我们自定义的函数即可
  • 其中,就需要用到我们接下来要用到的 signal函数(设置信号处理程序)
【2】信号捕捉函数signal语法介绍
  • signal函数: 用于设置信号处理程序——>当某个信号到达时,操作系统 应该调用哪个函数来处理该信号
代码语言:javascript代码运行次数:0运行复制
#include <signal.h>  //头文件
  
typedef void (*sighandler_t)(int);  
  
sighandler_t signal(int signum, sighandler_t handler);
参数:
     signum:这是一个整数,表示要处理的信号
     handler:这是一个指向函数的指针,该函数用于处理指定的信号
返回值:
     成功时,返回以前的信号处理程序的指针。
     失败时,返回SIG_ERR,并设置errno以指示错误。
【3】代码实现:设置信号的【默认处理函数】变成【自定义函数】

演示:

  • 原本2号信号对应
  • 如下图所示,经过signal函数处理后,2号信号发送给了handler函数
代码语言:javascript代码运行次数:0运行复制
#include <stdio.h>
#include <signal.h>
void handler(int sig)
{
       printf("catch a sig : %d\n", sig);
}
int main()
{
       signal(2, handler); //iganl函数就是来进行信号捕捉的
       while(1);
       return 0;
}
[hb@localhost code_test]$ ./sig 
^Ccatch a sig : 2
^Ccatch a sig : 2
^Ccatch a sig : 2
^Ccatch a sig : 2
^\Quit (core dumped)
[hb@localhost code_test]$
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。 原始发表:2024-10-23,如有侵权请联系 cloudcommunity@tencent 删除PCBlinux程序函数进程

本文标签: Linux解析在进程PCB中是如何实现信号的处理方式(抵达未决阻塞)