Linux系统编程(信号处理 sigacation函数和sigqueue函数 )

2023-07-14 01:30:00


前言

本篇文章我们来介绍一下sigacation函数和sigqueue函数。

一、sigaction

sigaction 是一个用于设置和检查信号处理程序的函数。它允许我们指定信号的处理方式,包括指定一个函数作为信号处理程序、设置标志位以及指定信号处理程序执行期间的信号屏蔽字等信息。

下面是 sigaction 函数的原型:

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

signum:要操作的信号的编号。可以是 POSIX 信号常量(如 SIGINT、SIGTERM 等)或自定义信号的编号。

act:一个指向 struct sigaction 结构的指针,包含了要设置的信号的处理方式和信号标志等信息。

oldact:一个可选的指向 struct sigaction 结构的指针。如果不为 NULL,将返回之前的信号处理程序的信息。

struct sigaction 结构体是用来描述信号的处理方式和信号标志等信息的,其定义如下:

struct sigaction {
    void (*sa_handler)(int);                  // 信号处理程序的函数指针
    sigset_t sa_mask;                          // 信号屏蔽字
    int sa_flags;                              // 信号标志
    void (*sa_sigaction)(int, siginfo_t *, void *);   // 使用 sa_sigaction 代替 sa_handler
};

sa_handler:一个函数指针,指向信号处理程序的函数。可以是一个用户定义的函数,也可以是特殊的值 SIG_DFL(默认处理程序)或 SIG_IGN(忽略处理程序)。

sa_mask:一个信号集,用来指定在信号处理程序执行期间要屏蔽的信号。它影响信号处理程序的上下文,可以通过 sigaddset、sigemptyset 等函数进行设置。

sa_flags:一组标志位,用于指定信号处理程序的行为。常见的标志位包括 SA_RESTART(在被信号中断的系统调用后自动重启)和 SA_NODEFER(在调用信号处理程序时不将本信号添加到进程的信号屏蔽字中)等。

sa_sigaction:一个函数指针,用于指定信号的扩展处理程序。它支持更多的参数,包括一个 siginfo_t 类型的结构体和一个 void * 类型的指针,用于传递关于信号的更多信息。

通过使用 sigaction 函数,我们可以设置信号处理程序的行为,包括调用用户定义的函数、忽略信号、重新启动系统调用等,并可以设置信号屏蔽字,以控制信号传递的方式。这使得我们可以对不同的信号进行个性化的处理。

siginfo_t 结构体是在信号处理程序中提供有关接收到信号的更详细信息的数据结构。它定义在 <signal.h> 头文件中,用于在处理程序中获取有关信号的附加信息。

下面是 siginfo_t 结构体的基本定义:

typedef struct siginfo {
    int si_signo;             // 信号编号
    int si_errno;             // 相关错误代码
    int si_code;              // 附加信号码
    pid_t si_pid;             // 发送信号的进程ID
    uid_t si_uid;             // 发送信号的用户ID
    void *si_addr;            // 引起信号的内存地址
    int si_status;            // 进程状态(仅在 SIGCHLD 信号中使用)
    long si_band;             // 事件发生的条件(仅在 SIGPOLL 信号中使用)
    struct sigval si_value;   // 附加的数据值
    union sigval si_utime;    // 用户CPU时间的计数值(仅在 SIGCHLD 信号中使用)
    union sigval si_stime;    // 系统CPU时间的计数值(仅在 SIGCHLD 信号中使用)
} siginfo_t;

二、sigqueue函数

sigqueue 函数用于向指定的进程发送特定的信号,并且可以传递一个额外的数据值。它提供了比 kill 函数更丰富的功能,可以用于进程间的高级通信。

以下是 sigqueue 函数的基本定义:

int sigqueue(pid_t pid, int sig, const union sigval value);

pid 是接收信号的进程的进程ID。
sig 是要发送的信号的编号,可以是标准信号(如 SIGTERM,SIGINT 等)或用户定义的信号。
value 是一个 union sigval 类型的结构体,用于传递附加的数据值。

使用 sigqueue 函数发送信号的步骤如下:

1.创建一个 union sigval 结构体并设置相应的值,以便将额外的数据传递给接收进程。

2.调用 sigqueue 函数,将目标进程的进程ID、信号编号和数据值作为参数传递给它。

3.如果成功发送信号,sigqueue 函数将返回0;如果出错,返回-1,并设置相应的错误码。

通过使用 sigqueue 函数发送带有附加数据的信号,接收进程可以从信号处理程序中获取这些附加数据,并根据需要采取相应的操作。这对于进程间的通信很有用,可以用于传递消息、触发特定的操作,或者向其他进程传递数据。

需要注意的是,接收进程必须正确设置信号处理程序来处理通过 sigqueue 函数发送的信号,并正确解析读取附加数据。

三、代码示例

接收信号处理函数:

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

void signal_Handle(int sig, siginfo_t* info, void* ucontext)
{
    printf("handler : sig = %d\n", sig);
    printf("handler : info->si_signo = %d\n", info->si_signo);
    printf("handler : info->si_code = %d\n", info->si_code);
    printf("handler : info->si_pid = %d\n", info->si_pid);
    printf("handler : info->si_value = %d\n", info->si_value.sival_int);
}

int main(int argc, char** argv)
{
    printf("pid :%d\n", getpid());

    struct sigaction act = {0};

    act.sa_sigaction = signal_Handle;
    act.sa_flags = SA_RESTART | SA_SIGINFO;

    /* 添加信号屏蔽字 */
    /* 下面信号在信号处理程序执行时会被暂时阻塞 */
    sigaddset(&act.sa_mask, 40);
    sigaddset(&act.sa_mask, SIGINT);

    /* 设置信号的处理行为,设置后40和SIGINT信号将由act里面的信号处理函数处理 */
    sigaction(40, &act, NULL);
    sigaction(SIGINT, &act, NULL);

    while(1)
    {
        sleep(1);
    }

    return 0;
}

发送信号处理函数:

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


int main(int argc, char** argv)
{
    pid_t pid = atoi(argv[1]);

    union sigval sv = {123456};
    
    sigqueue(pid, 40, sv);

    raise(SIGINT); 

    return 0;
}

运行结果:

从结果中我们可以看出接收到信号后打印出了对应的数据。
在这里插入图片描述

总结

本篇文章主要讲解了sigacation函数和sigqueue函数,相比于signal和kill函数,使用这两个函数来发送信号是比较灵活的。

更多推荐

编程小白如何学习RPA,0基础学习RPA攻略!

对于编程小白来说,学习RPA(机器人流程自动化)可能会感到有些无从下手。然而,只要按照一定的步骤和策略进行学习,从0开始掌握RPA并非难事。一、了解RPA的基本概念和优势在开始学习RPA之前,首先需要了解RPA的基本概念、特点和优势。RPA是一种使用自动化工具模拟人类在计算机上执行操作的技术,它可以帮助企业实现重复性、

第36章_瑞萨MCU零基础入门系列教程之步进电机控制实验

本教程基于韦东山百问网出的DShanMCU-RA6M5开发板进行编写,需要的同学可以在这里获取:https://item.taobao.com/item.htm?id=728461040949配套资料获取:https://renesas-docs.100ask.net瑞萨MCU零基础入门系列教程汇总:https://b

Spring Boot 简介与入门

🌷🍁博主猫头虎带您GotoNewWorld.✨🍁🦄博客首页——猫头虎的博客🎐🐳《面试题大全专栏》文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺🌊《IDEA开发秘籍专栏》学会IDEA常用操作,工作效率翻倍~💐🌊《100天精通Golang(基础入门篇)》学会Golang语言,畅玩云原生,走遍大

数据结构:树和二叉树之-堆排列 (万字详解)

目录树概念及结构1.1树的概念1.2树的表示​编辑2.二叉树概念及结构2.1概念2.2数据结构中的二叉树:​编辑2.3特殊的二叉树:​编辑2.4二叉树的存储结构2.4.1顺序存储:2.4.2链式存储:二叉树的实现及大小堆排列1功能展示2定义基本结构3初始化4打印5销毁6插入7向上调整8交换两数组元素之间的值9删除10向

Kubernetes(K8S)集群部署

目录一、创建3台虚拟机二、为每台虚拟机安装Docker三、安装kubelet3.1安装要求3.2为每台服务器完成前置设置3.3为每台服务器安装kubelet、kubeadm、kubectl四、使用kubeadm引导集群4.1master服务器4.2node1、node2服务器4.3初始化主节点4.4work节点加入集群

【C语言趣味教程】(1) 深入浅出 HelloWorld:通过 HelloWorld 展开教学 | 头文件详解 | main 函数详解

🔗《C语言趣味教程》👈猛戳订阅!!!💭写在前面:这是一套C语言趣味教学专栏,目前正在火热连载中,欢迎猛戳订阅!本专栏保证篇篇精品,继续保持本人一贯的幽默式写作风格,当然,在有趣的同时也同样会保证文章的质量,旨在能够产出"有趣的干货"!本系列教程不管是零基础还是有基础的读者都可以阅读,可以先看看目录!标题前带星号(

第二证券:什么是a股b股?

在我国的股市中,我们经常会听到“A股”和“B股”这两个名词。那么,终究什么是A股和B股呢?首先,A股全称为“A股票”,是指在我国境内上市的以人民币计价的股票。A股首要面向国内出资者,只要具有必定条件的内地居民和安排出资者才可出资A股。此外,A股还分为两种:人民币A股和港币A股。前者是在我国境内发行的A股,后者则是在香港

Redis 面试题——缓存穿透、缓存击穿和缓存雪崩

目录1.缓存穿透2.缓存击穿3.缓存雪崩4.总结参考文章:缓存实战(1)缓存雪崩、缓存击穿和缓存穿透入门简介及解决方案1.缓存穿透(1)问题描述:缓存穿透是指在高并发场景下,大量的请求访问一个不存在于缓存中也不存在于数据库中的数据,导致每次请求都要查询数据库,增加了数据库的负载。通常发生在恶意攻击、频繁访问不存在的数据

OSCP系列靶场-Esay-Moneybox保姆级

OSCP系列靶场-Esay-Moneybox目录OSCP系列靶场-Esay-Moneybox总结准备工作信息收集-端口扫描目标开放端口收集目标端口对应服务探测信息收集-端口测试21-FTP端口的信息收集21-FTP版本版本信息21-FTP端口匿名登录测试(存在)21-FTP端口-文件GET收集21-FTP端口-PUT上

微服务 第一章 Java线程池技术应用

系列文章目录第一章Java线程池技术应用文章目录系列文章目录@[TOC](文章目录)前言1、Java创建线程方式回顾1.1、继承Thread类(只运行一次)1.1.1、改造成主线程常驻,每秒开启新线程运行1.1.2、匿名内部类1.1.3、缺点1.1.4、扩展知识:Java内部类1.1.4.1、静态内部类1.1.4.2、

机器学习实战:Python基于Ridge岭回归进行正则化(十三)

文章目录1.前言1.1岭回归的介绍1.2岭回归的应用2.自定义数据集实战演示2.1导入函数2.2创建数据集2.3alpha=0、1、10、100的分别情况3.Dushanbe_house数据集实战演示3.1导入函数和数据3.2剔除空值及可视化3.3整理数据3.4训练和测试数据集3.5评估数据集4.讨论1.前言1.1岭回

热文推荐