信号、signal 函数、sigaction 函数

article/2025/10/16 8:05:34

文章目录

  • 1.信号的基本概念
  • 2.利用 kill 命令发送信号
  • 3.信号处理的相关动作
  • 4.信号与 signal 函数
    • 4.1 signal 函数示例一
    • 4.2 signal 函数示例二
  • 5.利用 sigaction 函数进行信号处理
  • 6.利用信号处理技术消灭僵尸进程

1.信号的基本概念

发送信号是进程之间常用的通信手段。信号用来通知某个进程发生了某一个事情,事情、信号都是突发事件,信号是异步发生的,信号也被称为“软件中断”。

信号如何产生:

  • 某个进程发送给另外一个进程或者发送给自己
  • 由内核发送给某个进程
    • 通过在键盘上输入命令,如 Ctrl + Ckill
    • 内存访问异常、除数为 0 0 0 等等,硬件都会检测到并且通知内核

UNIX 以及类 UNIX 操作系统支持的信号数量各不相同。信号有名字,都是以 SIG 开头,如终端断开信号 SIGHUP。其实,信号就是一些宏定义的正整数常量(从数字 1 1 1 开始)。

查找 signal.hSIGHUP 的命令如下:

sudo find / -name "signal.h" | xargs grep -in "SIGHUP"

在这里插入图片描述

2.利用 kill 命令发送信号

创建一个 test.c 文件,其内容如下:

#include <stdio.h>
#include <unistd.h>int main(int argc, char* const* argv)
{printf("你好,世界!\n");for (;;){sleep(1);printf("休息1秒\n");}printf("程序退出,再见!\n");return 0;
}

编译 test.c 的命令如下:

gcc test.c -o test

运行 test 的命令如下:

./test

查看 bash 进程和 test 进程的命令如下:

ps -eo pid,ppid,pgid,sid,tty,comm | grep -E 'bash|PID|test'

在这里插入图片描述

跟踪 test 进程的命令如下:

sudo strace -e trace=signal -p 1268

kill 1268 命令就是往 test 进程发送 SIGTERM 终止信号:

在这里插入图片描述

kill -2 1365 命令就是往 test 进程发送 SIGINT 中断信号:

在这里插入图片描述

关于 kill 命令及 Linux 系统支持的部分信号:https://wker.com/linux-command/kill.html

3.信号处理的相关动作

上面提到的 kill 命令只是发个信号,而不是单纯的杀死的意思。

当某个信号出现时,我们可以按三种方式之一进行处理:

  • 执行系统默认动作,绝大多数信号的默认动作是杀死这个进程;
  • 忽略该信号;
  • 捕捉该信号,即自己写个处理函数,当信号来的时候,就调用处理函数来处理。

注意:SIGKILLSIGSTOP 信号既不能被忽略,也不能被捕捉。

4.信号与 signal 函数

进程:“嘿,操作系统!如果我之前创建的子进程终止,就帮我调用 zombie_handler 函数。”

操作系统:“好的!如果你的子进程终止,我会帮你调用 zombie_handler 函数,你先把该函数要执行的语句编好!”

上述对话中进程所讲的相当于“注册信号”过程,即进程发现自己的子进程结束时,请求操作系统调用特定函数。该请求通过 signal 函数调用完成,因此称 signal 函数为信号注册函数。

#include <signal.h>void (*signal(int signo, void (*func)(int)))(int);// 为了在产生信号时调用,返回之前注册的函数指针
// 函数名:signal
// 参数:int signo, void(*func)(int)
// 返回类型:参数为int型,返回void型函数指针

调用上述函数时,第一个参数为特殊情况信息,第二个参数为特殊情况下将要调用的函数的地址值(指针)。发生第一个参数代表的情况时,调用第二个参数所指的函数。

// 子进程终止则调用mychild函数
signal(SIGCHLD, mychild);
// 常数SIGCHLD定义了子进程终止的情况,应成为signal函数的第一个参数
// 此时mychild函数的参数应为int,返回值类型应为void,只有这样才能成为signal函数的第二个参数
// 已到通过alarm函数注册的时间,请调用timeout函数
signal(SIGALRM, timeout);
// 输入CTRL+C时调用keycontrol函数
signal(SIGINT, keycontrol);

以上就是信号注册过程。注册好信号后,发生注册信号时(注册的情况发生时),操作系统将调用该信号对应的函数。

4.1 signal 函数示例一

下面首先介绍 alarm 函数。

#include <unistd.h>unsigned int alarm(unsigned int seconds);// 返回0或以秒为单位的距SIGALRM信号发生所剩时间

如果调用该函数的同时向它传递一个正整型参数,相应时间后(以秒为单位)将产生 SIGALRM 信号。若向该函数传递 0 0 0,则之前对 SIGALRM 信号的预约将取消。如果通过该函数预约信号后未指定该信号对应的处理函数,则(通过调用 signal 函数)终止进程,不做任何处理。

接下来给出信号处理相关示例。

#include <stdio.h>
#include <unistd.h>
#include <signal.h>// 定义信号处理函数,这种类型的函数称为信号处理器(Handler)
void timeout(int sig)
{if (sig == SIGALRM)puts("Time out!");// 为了每隔2秒重复产生SIGALRM信号,在信号处理器中调用alarm函数alarm(2);
}// 定义信号处理函数,这种类型的函数称为信号处理器(Handler)
void keycontrol(int sig)
{if (sig == SIGINT)puts("CTRL+C pressed");
}int main(int argc, char *argv[])
{int i;// 注册SIGALRM、SIGINT信号及相应处理器signal(SIGALRM, timeout);signal(SIGINT, keycontrol);// 预约2秒后发生SIGALRM信号alarm(2);// 为了查看信号产生和信号处理器的执行,提供每次100秒、共3次的等待时间,在循环中调用sleep函数。// 也就是说,再过300秒、约5分钟后终止程序,这是相当长的一段时间,但实际执行时只需不到10秒。for (i = 0; i < 3; i++){puts("wait...");sleep(100);}return 0;
}

编译运行:

gcc signal.c -o signal
./signal

输出结果:

在这里插入图片描述

上述是没有任何输入时的运行结果。

下面在运行过程中输入 CTRL+C,可以看到输出“CTRL+C pressed”字符串。

在这里插入图片描述

有一点必须说明:“发生信号时将唤醒由于调用 sleep 函数而进入阻塞状态的进程。”

调用函数的主体的确是操作系统,但进程处于睡眠状态时无法调用函数。因此,产生信号时,为了调用信号处理器,将唤醒由于调用 sleep 函数而进入阻塞状态的进程。而且,进程一旦被唤醒,就不会再进入睡眠状态。即使还未到 sleep 函数中规定的时间也是如此。所以,上述示例运行不到 10 10 10 秒就会结束,连续输入 CTRL+C 则有可能 1 1 1 秒都不到。

4.2 signal 函数示例二

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>// 信号处理函数
void sig_usr(int signo)
{if (signo == SIGUSR1){printf("收到了SIGUSR1信号!\n");}else if (signo == SIGUSR2){printf("收到了SIGUSR2信号!\n");}else{printf("收到了未捕捉的信号%d!\n", signo);}
}int main(int argc, char* const* argv)
{// 系统函数,第一个参数是个信号,第二个参数是个函数指针,代表一个针对该信号的捕捉处理函数if (signal(SIGUSR1, sig_usr) == SIG_ERR){printf("无法捕捉SIGUSR1信号!\n");}if (signal(SIGUSR2, sig_usr) == SIG_ERR){printf("无法捕捉SIGUSR2信号!\n");}for (;;){sleep(1);printf("休息1秒\n");}printf("再见!\n");return 0;
}

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

5.利用 sigaction 函数进行信号处理

sigaction 函数,它类似于 signal 函数,而且完全可以代替 signal 函数,也更稳定。之所以稳定,是因为:“signal 函数在 UNIX 系列的不同操作系统中可能存在区别,但 sigaction 函数完全相同。”

实际上现在很少使用 signal 函数编写程序,它只是为了保持对旧程序的兼容。下面介绍 sigaction 函数,但只讲解可替换 signal 函数的功能,因为全面介绍会给各位带来不必要的负担。

#include <signal.h>int sigaction(int signo, const struct sigaction *act, struct sigaction *oldact);// 成功时返回0,失败时返回-1。
// signo:与signal函数相同,传递信号信息。
// act:对应于第一个参数的信号处理函数(信号处理器)信息。
// oldact:通过此参数获取之前注册的信号处理函数指针,若不需要则传递0。

声明并初始化 sigaction 结构体变量以调用上述函数,该结构体定义如下:

struct sigaction
{void (*sa_handler)(int);sigset_t sa_mask;int sa_flags;
};

此结构体的 sa_handler 成员保存信号处理函数的指针值(地址值)。sa_mask 和 sa_flags 的所有位均初始化为 0 0 0 即可。这 2 2 2 个成员用于指定信号相关的选项和特性,而我们的目的主要是防止产生僵尸进程,故省略。

#include <stdio.h>
#include <unistd.h>
#include <signal.h>// #define _XOPEN_SOURCE 700void timeout(int sig)
{if (sig == SIGALRM)puts("Time out!");alarm(2);
}int main(int argc, char *argv[])
{int i;// 为了注册信号处理函数,声明sigaction结构体变量并在sa_handler成员中保存函数指针值struct sigaction act;act.sa_handler = timeout;// 调用sigemptyset函数将sa_mask成员的所有位初始化为0sigemptyset(&act.sa_mask);// sa_flags成员同样初始化为0act.sa_flags = 0;// 注册SIGALRM信号的处理器。调用alarm函数预约2秒后发生SIGALRM信号sigaction(SIGALRM, &act, 0);alarm(2);for (i = 0; i < 3; i++){puts("wait...");sleep(100);}return 0;
}

编译运行:

gcc sigaction.c -o sigaction
./sigaction

输出结果:

在这里插入图片描述

6.利用信号处理技术消灭僵尸进程

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>// #define _XOPEN_SOURCE 700void read_childproc(int sig)
{int status;pid_t id = waitpid(-1, &status, WNOHANG);if (WIFEXITED(status)){printf("Removed proc id: %d\n", id);printf("Child send: %d\n", WEXITSTATUS(status));}
}int main(int argc, char *argv[])
{pid_t pid;// 注册SIGCHLD信号对应的处理器。若子进程终止,则调用第7行中定义的函数。// 处理函数中调用了waitpid函数,所以子进程将正常终止,不会成为僵尸进程。struct sigaction act;act.sa_handler = read_childproc;sigemptyset(&act.sa_mask);act.sa_flags = 0;sigaction(SIGCHLD, &act, 0);// 创建子进程pid = fork();if (pid == 0)    // 子进程执行区域{puts("Hi! I'm child process");sleep(10);return 12;}else    // 父进程执行区域{printf("Child proc id: %d\n", pid);// 创建子进程pid = fork();if (pid == 0)    // 另一个子进程执行区域{puts("Hi! I'm child process");sleep(15);exit(24);}else{int i;printf("Child proc id: %d\n", pid);// 为了等待发生SIGCHLD信号,使父进程共暂停5次,每次间隔5秒。// 发生信号时,父进程将被唤醒,因此实际暂停时间不到25秒。for (i = 0; i < 5; i++){puts("wait...");sleep(5);}}}return 0;
}

编译运行:

gcc remove_zombie.c -o zombie
./zombie

输出结果:

在这里插入图片描述

可以看出,子进程并未变成僵尸进程,而是正常终止了。


http://chatgpt.dhexx.cn/article/h24ToAv9.shtml

相关文章

linux sigaction详解

参看文档&#xff1a; https://blog.csdn.net/weixin_43743847/article/details/90299204https://blog.csdn.net/u010150046/article/details/77344438https://bbs.csdn.net/topics/370255407 一&#xff1a;函数原型介绍 int sigaction(int signum, const struct sigaction …

sigaction函数

#include <signal.h> int sigaction(int signum, const struct sigaction *act,struct sigaction *oldact); //The sigaction() system call is used to change the action taken by a process on receipt of a specific signal. sigaction()系统调用用于改变进程在接收…

c语言sigaction,c语言信号处理sigaction

c语言信号处理sigaction (2011-04-18 23:45:19) 标签: c语言 信号处理 sigaction sighup it 分类: c 信号安装函数sigaction(int signum,const struct sigaction *act,struct sigaction *oldact)的第二个参数是一个指向sigaction结构的指针(结构体名称与函数名一样,千万别弄…

linux中sigaction函数详解

一、函数原型&#xff1a;sigaction函数的功能是检查或修改与指定信号相关联的处理动作&#xff08;可同时两种操作&#xff09; int sigaction(int signum, const struct sigaction *act,struct sigaction *oldact);signum参数指出要捕获的信号类型&#xff0c;act参数指定新…

Qt线程QThread详解

目录 前言1.QThread介绍2.QThread示例一3.QThread示例二4.线程同步 前言 在程序中使用线程可以提高程序的性能、并发性、响应性和稳定性&#xff0c;使得程序设计更加灵活和简单。但是&#xff0c;线程编程也有一些挑战&#xff0c;如线程安全性和死锁等问题需要格外注意。我们…

PyQT5 多线程 QThread

PyQT5 多线程 在常规的界面软件中&#xff0c;需要将UI线程和工作线程加以区分&#xff0c;主要原因是某些工作线程很复杂且耗时&#xff0c;比如下载某个文件或者长时间的计算&#xff0c;当执行这些进程时&#xff0c;UI主进程会被阻塞&#xff0c;界面会出现未响应的状态&a…

QThread之moveToThread用法

一、怎么用 使用一个QObject作为Worker&#xff0c;并moveToThread到线程上&#xff0c;那么这个QObject生存在此线程上&#xff0c;其信号会在此线程上发射&#xff0c;其槽函数在此线程上执行。 意味着什么&#xff0c;意味着多线程操作时&#xff0c;若通过信号槽方式&…

Qt 多线程编程的 QThread 类 (详细)

本文结构如下&#xff1a; 概述优雅的开始我们的多线程编程之旅 我们该把耗时代码放在哪里&#xff1f;再谈 moveToThread()启动线程前的准备工作 开多少个线程比较合适&#xff1f;设置栈大小启动线程/退出线程 启动线程优雅的退出线程操作运行中的线程 获取状态 运行状态线程…

QT之多线程(QThread)的简单使用

一、线程简述 线程&#xff08;thread&#xff09;是操作系统能够进行运算调度的最小单位。一条线程指的是进程中一个单一顺序的控制流&#xff0c;它被包含在进程之中&#xff0c;是进程中的实际运作单位。一个进程中可以并发多个线程&#xff0c;每条线程并行执行不同的任务…

Qt 之 QThread(深入理解)

作者: 一去、二三里 个人微信号: iwaleon 微信公众号: 高效程序员 为了让程序尽快响应用户操作,在开发应用程序时经常会使用到线程。对于耗时操作如果不使用线程,UI界面将会长时间处于停滞状态,这种情况是用户非常不愿意看到的,我们可以用线程来解决这个问题。 前面,已…

Qt线程:QThread

一、描述 一个QThread对象管理程序内的一个线程&#xff0c;QThreads在run()中开始执行。默认情况下&#xff0c;run()通过调用exec()启动事件循环&#xff0c;并在线程内部运行一个Qt事件循环。 可以通过使用 QObject::moveToThread() 将对象移动到线程来使用它们。 class W…

PyQt中的多线程QThread示例

PyQt中的多线程 一、PyQt中的多线程二、创建线程2.1 设计ui界面2.2 设计工作线程2.3 主程序设计 三、运行结果示例 一、PyQt中的多线程 传统的图形用户界面应用程序都只有一个执行线程&#xff0c;并且一次只执行一个操作。如果用户从用户界面中调用一个比较耗时的操作&#x…

【Qt】Qt的线程(两种QThread类的详细使用方式)

Qt提供QThread类以进行多任务处理。与多任务处理一样&#xff0c;Qt提供的线程可以做到单个线程做不到的事情。例如&#xff0c;网络应用程序中&#xff0c;可以使用线程处理多种连接器。 QThread继承自QObject类&#xff0c;且提供QMutex类以实现同步。线程和进程共享全局变量…

Qt - 一文理解QThread多线程(万字剖析整理)

目录 为什么需要多线程QThread使用方法new QThread Class & Override run()new Object Class & moveToThread(new QThread) connect事件循环源码分析如何正确退出线程堆栈大小优先级线程间通讯线程同步互斥锁读写锁信号量条件变量 可重入与线程安全QObject的可重入性开…

Qt 线程中QThread的使用

文章目录 Qt 线程中QThread的使用1. 线程类 QThread1.1 常用共用成员函数1.2 信号槽1.3 静态函数1.4 任务处理函数 2. 使用方式 12.2 示例代码3. 使用方式 23.1 操作步骤3.2 示例代码 Qt 线程中QThread的使用 在进行桌面应用程序开发的时候&#xff0c; 假设应用程序在某些情况…

Qt之QThread(深入理解)

简述 为了让程序尽快响应用户操作&#xff0c;在开发应用程序时经常会使用到线程。对于耗时操作如果不使用线程&#xff0c;UI界面将会长时间处于停滞状态&#xff0c;这种情况是用户非常不愿意看到的&#xff0c;我们可以用线程来解决这个问题。 前面&#xff0c;已经介绍了…

Qt之QThread介绍(常用接口及实现、自动释放内存、关闭窗口时停止线程运行、同步互斥)

在程序设计中&#xff0c;为了不影响主程序的执行&#xff0c;常常把耗时操作放到一个单独的线程中执行。Qt对多线程操作有着完整的支持&#xff0c;Qt中通过继承QThread并重写run()方法的方式实现多线程代码的编写。针对线程之间的同步与互斥问题&#xff0c;Qt还提供了QMutex…

Qt线程QThread开启和安全退出

1、线程开启 Qt中&#xff0c;开启子线程&#xff0c;一般有两种方法&#xff1a; a, 定义工作类worker: worker继承 QThread, 重写run函数&#xff0c;在主线程中实例化worker&#xff0c;把耗时工作放进worker的run函数中完成&#xff0c;结束后&#xff0c;往主线程中发信…

QThread的用法

概述 QThread类提供了一个与平台无关的管理线程的方法。一个QThread对象管理一个线程。QThread的执行从run()函数的执行开始&#xff0c;在Qt自带的QThread类中&#xff0c;run()函数通过调用exec()函数来启动事件循环机制&#xff0c;并且在线程内部处理Qt的事件。在Qt中建立线…

Oracle 定时任务执行存储过程【建议收藏】

首先用一个完整的例子来实现定时执行存储过程。 任务目标&#xff1a;每小时向test表中插入一条数据。 实现方案&#xff1a; 1.通过 oracle 中 dbms_job 完成存储过程的定时调用 2.在存储过程中完成相应的逻辑操作 实现步骤&#xff1a; 1.创建一个测试表 create table test…