SIGINT SIGPIPE SIGTERM SIGSEGV SIG_IGN产生原因及处理

本文介绍了在进程通信中如何处理不同类型的信号,如SIGINT(Ctrl+C产生的)、SIGPIPE(socket通信时A关闭连接导致B收到的)和SIGTERM(kill命令发送的)。当接收到这些信号时,通常会终止进程,但可以通过自定义处理函数来执行清理操作,如保存数据和关闭文件。SIGSEGV则是在尝试访问非法内存时触发,也会导致进程终止。通过设置SIG_IGN,可以忽略某些信号,例如SIGPIPE。示例代码展示了如何注册信号处理函数来优雅地处理这些情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

SIGINT:由Ctr+C产生
SIGPIPE:A与B进行socket通信,假如A关闭了socket,这个时候B依然write信息发送,当第二次write时,B本机会给这个进程发送一个SIGPIPE信息,SIGPIPE默认的作用是终止这个进程,如果不打算终止的话,那么要自定义对这个消息的处理函数。
SIGTERM是kill或killall命令发送到进程的默认信号,它会导致进程终止,如果要在结束之前做些其他的操作,比如保存数据,关闭文件,那么可以自定义对这个信号的处理函数。
SIGSEGV:试图访问未分配给自己的内存, 或试图往没有写权限的内存地址写数据时,系统就会发送给进程这样的信号,会报段错误,并终止进程,如果想在终止前做一些操作需要自定义对这个信号的处理。
SIG_IGN:忽略信号,可以把其他信号设置为这个信号,这样被设置的信号就会忽略,但不是所有的信号都可以设置为SIG_IGN信号。如调用了signal(SIGPIPE, SIG_IGN), 这样产生 SIGPIPE 信号时就不会中止程序,直接把这个信号忽略掉。
void signal_handler(int sig)
{
//在这里做一些进程终止之前的操作,比如保存数据到文件,关闭文件,主动终止已启动的线程等。
}
void init_signal()
{
signal(SIGINT, signal_handler);
signal(SIGPIPE, SIG_IGN);
signal(SIGTERM, signal_handler);
signal(SIGSEGV, signal_handler);
}

#include <signal.h>
int main()
{
//注意这个一般都放在man()函数的靠前位置,在问题发生之前,提前注册好,当发生后就按自定义的方式处理。
init_signal();
//开始写代码。。。。
}

a) 要编写一个简单的进程间软中断通信程序,我们可以利用Unix/Linux系统提供的信号处理机制。这里我会给出一个基于C语言的例子: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/wait.h> #include <signal.h> // 定义一个接收信号并打印信息的函数 void handle_signal(int signum) { switch (signum) { case SIGUSR1: printf("Received signal USR1 from parent process.\n"); break; default: printf("Unexpected signal %d received.\n", signum); } } int main() { // 创建子进程 pid_t child_pid = fork(); if (child_pid == 0) { // 子进程 // 注册信号处理signal(SIGUSR1, handle_signal); // 进入死循环等待外部中断 while (1) { sleep(1); // 每秒检查一次信号 } } else { // 父进程 // 发送信号给子进程 kill(child_pid, SIGUSR1); // 等待子进程结束 wait(NULL); // 修改信号处理策略,忽略后续的SIGINTSIGQUIT signal(SIGINT, SIG_IGN); signal(SIGQUIT, SIG_IGN); // 再次发送信号,这将被忽略 kill(child_pid, SIGINT); // 或者 kill(child_pid, SIGQUIT); kill(child_pid, SIGUSR1); // 只有SIGUSR1会被处理 } return 0; } ``` b) 当你在父进程中设置了`signal(SIGINT, SIG_IGN)`和`signal(SIGQUIT, SIG_IGN)`之后,当你按下Ctrl+C或者按Q退出程序时,由于我们已经忽略了这两个信号(SIGINT代表Ctrl+C,SIGQUIT代表强制退出),所以子进程不会收到这些信号,也不会有任何反应。只有当父进程向子进程发送SIGUSR1信号时,子进程才会响应。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

、、、、南山小雨、、、、

分享对你有帮助,打赏一下吧!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值