一、什么是信号?

信号就像是一个突然的电话铃声,它会打断正在进行的程序并引起其注意。

在Linux系统中,信号是一种软件中断,它通常是异步发生的,可以用来通知进程某个事件已经发生。。每个信号都有一个唯一的编号,编号从1开始。进程可以通过注册信号处理函数来处理信号。

二、信号的分类

Linux系统中的信号有两类:标准信号和实时信号。

  • 标准信号是传统Unix系统中的信号,编号范围从1到31。
  • 实时信号是Linux独有的信号,编号范围从32到64。

三、信号的使用

1、注册信号处理函数

在C语言中,可以使用signal函数来注册信号处理函数。signal函数原型如下:

void (*signal(int signum, void (*handler)(int)))(int);

其中,signum参数表示要注册的信号编号,handler参数表示信号处理函数。signal函数会返回上一次注册的信号处理函数的地址。

下面是一个简单的例子,注册SIGINT信号的处理函数:

#include <stdio.h>
#include <signal.h>
#include <stdlib.h>
 
void sigint_handler(int signo)
{
    printf("Received SIGINT signal!\n");
    exit(0);
}
 
int main()
{
    if (signal(SIGINT, sigint_handler) == SIG_ERR) {
        perror("signal");
        exit(1);
    }
 
    while (1) {
        printf("Hello World!\n");
        sleep(1);
    }
 
    return 0;
}

在上面的例子中,我们首先定义了一个名为sigint_handler的函数,用于处理SIGINT信号。

在main函数中,我们使用signal函数注册了SIGINT信号的处理函数。如果signal函数返回的值是SIG_ERR,表示注册信号处理函数失败。

在循环中,我们只是简单地输出Hello World!字符串,并使用sleep函数暂停1秒钟,以便观察信号处理过程。

  • 编译并运行,打印如下
[wayne@wayne:~]./signal
Hello World!
Hello World!
Hello World!

2、发送信号

2.1 kill函数

在Linux系统中,可以使用kill函数向进程发送信号。kill函数原型如下:

int kill(pid_t pid, int sig);

其中,pid参数表示进程的PID号,sig参数表示要发送的信号编号。如果pid参数的值是0,表示将信号发送给与当前进程属于同一个进程组的所有进程。

下面是一个例子,向指定进程发送SIGINT信号:

#include <stdio.h>
#include <signal.h>
#include <stdlib.h>
#include <unistd.h>
 
int main(int argc, char **argv)
{
    if (argc < 2) {
        fprintf(stderr, "Usage: %s <pid>\n", argv[0]);
        exit(1);
    }

    pid_t pid = atoi(argv[1]);

    if (kill(pid, SIGINT) == -1) {
           perror("kill");
        exit(1);
    }

    return 0;
}

在上面的例子中,我们首先检查命令行参数,如果参数不足,就打印使用说明并退出程序。

然后我们将命令行参数转换为进程PID号,并使用kill函数向该进程发送SIGINT信号。如果kill函数返回-1,表示发送信号失败。

  • 编译并运行,打印如下
[wayne@wayne:~] ps aux | grep signal
wayne     5902  0.0  0.0   2028   280 pts/25   S+   19:04   0:00 ./signal
wayne     5904  0.0  0.0   6108   852 pts/24   S+   19:04   0:00 grep --color=auto signal

[wayne@wayne:~] ./killsignal 5902


// 此时上面运行的signal进程会收到信号,打印如下信息
Received SIGINT signal!

2.2 kill命令

我们也可以在shell中,用kill命令向进程发送信号,kill命令的语法如下:

kill [options] <pid>

其中,pid表示要发送信号的进程PID号。options是一些可选参数,例如,可以使用-9参数发送SIGKILL信号。

我们可以打开一个新的终端窗口,使用ps命令查看本文介绍的示例程序的PID号,然后使用kill命令向该进程发送SIGINT信号,示例如下:

$ ps aux | grep signal
wayne     5902  0.0  0.0   2028   280 pts/25   S+   19:04   0:00 ./signal
wayne     5904  0.0  0.0   6108   852 pts/24   S+   19:04   0:00 grep --color=auto signal

$ kill -2 5902

这里我们使用ps命令查找名为signal的进程的PID号为5902,然后使用kill命令向该进程发送SIGINT信号(信号编号为2)。在执行kill命令之后,我们可以看到示例程序输出了"Received SIGINT signal"这一行信息,说明信号处理函数已经被正确调用了。

四、常用信号

Linux系统中常用的信号有很多,下面列出一些常用的信号及其含义:

信号编号信号名称含义
1SIGHUP终端挂起或者控制进程结束
2SIGINT中断信号,通常是CTRL-C
3SIGQUIT终止信号,通常是CTRL-\
9SIGKILL强制终止进程
11SIGSEGV段错误
15SIGTERM终止信号,通常是kill命令发送的信号
18SIGCONT继续执行被暂停的进程
19SIGSTOP暂停进程
20SIGTSTP终端挂起或者CTRL-Z

五、小结

总的来说,Linux 信号是一种用于通知进程发生某个事件或错误的机制,可以用于处理异常情况、进程间通信等多种场景。

以上,如果觉得对你有帮助,点个赞再走吧,这样@知微之见也有更新下去的动力!

也欢迎私信我,一起交流!

作者:|知微之见|,原文链接: https://segmentfault.com/a/1190000043545265

文章推荐

APP中RN页面渲染流程-ReactNative源码分析

Java 泛型:理解和应用

8张图带你全面了解kafka的核心机制

.NET6项目连接数据库方式方法

一文梳理z-index和层叠上下文

Linux安装Net7SDK运行Net项目

ResNet50的猫狗分类训练及预测

Redis缓存高可用集群

Cesium官方教程——Fabric

c++算法竞赛常用板子集合(持续更新)

求职面试场景下Spring都有哪些完美回答?

怎样生成分布式的流水ID