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

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

前言

本篇文章我们来介绍一下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函数,使用这两个函数来发送信号是比较灵活的。


相关文章
|
18天前
|
存储 缓存 监控
Linux缓存管理:如何安全地清理系统缓存
在Linux系统中,内存管理至关重要。本文详细介绍了如何安全地清理系统缓存,特别是通过使用`/proc/sys/vm/drop_caches`接口。内容包括清理缓存的原因、步骤、注意事项和最佳实践,帮助你在必要时优化系统性能。
151 78
|
8天前
|
Linux
【Linux】System V信号量详解以及semget()、semctl()和semop()函数讲解
System V信号量的概念及其在Linux中的使用,包括 `semget()`、`semctl()`和 `semop()`函数的具体使用方法。通过实际代码示例,演示了如何创建、初始化和使用信号量进行进程间同步。掌握这些知识,可以有效解决多进程编程中的同步问题,提高程序的可靠性和稳定性。
47 19
|
22天前
|
Linux Shell 网络安全
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
本指南介绍如何利用 HTA 文件和 Metasploit 框架进行渗透测试。通过创建反向 shell、生成 HTA 文件、设置 HTTP 服务器和发送文件,最终实现对目标系统的控制。适用于教育目的,需合法授权。
54 9
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
|
10天前
|
Linux Android开发 开发者
linux m、mm、mmm函数和make的区别
通过理解和合理使用这些命令,可以更高效地进行项目构建和管理,特别是在复杂的 Android 开发环境中。
40 18
|
18天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
83 13
|
18天前
|
Ubuntu Linux C++
Win10系统上直接使用linux子系统教程(仅需五步!超简单,快速上手)
本文介绍了如何在Windows 10上安装并使用Linux子系统。首先,通过应用商店安装Windows Terminal和Linux系统(如Ubuntu)。接着,在控制面板中启用“适用于Linux的Windows子系统”并重启电脑。最后,在Windows Terminal中选择安装的Linux系统即可开始使用。文中还提供了注意事项和进一步配置的链接。
40 0
|
Linux
Linux下信号处理(发送、捕获)
Linux下进行应用编程时,信号的处理必不可少。信号可以用于多进程间通信,这篇文章就介绍如何发送信号,捕获信号。
487 0
|
2月前
|
Linux 网络安全 数据安全/隐私保护
Linux 超级强大的十六进制 dump 工具:XXD 命令,我教你应该如何使用!
在 Linux 系统中,xxd 命令是一个强大的十六进制 dump 工具,可以将文件或数据以十六进制和 ASCII 字符形式显示,帮助用户深入了解和分析数据。本文详细介绍了 xxd 命令的基本用法、高级功能及实际应用案例,包括查看文件内容、指定输出格式、写入文件、数据比较、数据提取、数据转换和数据加密解密等。通过掌握这些技巧,用户可以更高效地处理各种数据问题。
138 8
|
2月前
|
监控 Linux
如何检查 Linux 内存使用量是否耗尽?这 5 个命令堪称绝了!
本文介绍了在Linux系统中检查内存使用情况的5个常用命令:`free`、`top`、`vmstat`、`pidstat` 和 `/proc/meminfo` 文件,帮助用户准确监控内存状态,确保系统稳定运行。
555 6
|
2月前
|
Linux
在 Linux 系统中,“cd”命令用于切换当前工作目录
在 Linux 系统中,“cd”命令用于切换当前工作目录。本文详细介绍了“cd”命令的基本用法和常见技巧,包括使用“.”、“..”、“~”、绝对路径和相对路径,以及快速切换到上一次工作目录等。此外,还探讨了高级技巧,如使用通配符、结合其他命令、在脚本中使用,以及实际应用案例,帮助读者提高工作效率。
104 3