基于linux下的高并发服务器开发(第二章)- 2.24 信号集及相关函数

1.用户通过键盘 Ctrl + C, 产生2号信号SIGINT (信号被创建)

**2.信号产生但是没有被处理 (未决)

  • 在内核中将所有的没有被处理的信号存储在一个集合中 (未决信号集)
  • SIGINT信号状态被存储在第二个标志位上
  • 这个标志位的值为0, 说明信号不是未决状态
  • 这个标志位的值为1, 说明信号处于未决状态
    3.这个未决状态的信号,需要被处理,处理之前需要和另一个信号集(阻塞信号集),进行比较
  • 阻塞信号集默认不阻塞任何的信号
  • 如果想要阻塞某些信号需要用户调用系统的API**

**4.在处理的时候和阻塞信号集中的标志位进行查询,看是不是对该信号设置阻塞了

  • 如果没有阻塞,这个信号就被处理
  • 如果阻塞了,这个信号就继续处于未决状态,直到阻塞解除,这个信号就被处理**

以下信号集相关的函数都是对自定义的信号集进行操作。

int sigemptyset(sigset_t *set);

- 功能:清空信号集中的数据,将信号集中的所有的标志位置为0

- 参数:set,传出参数,需要操作的信号集

- 返回值:成功返回0, 失败返回-1

int sigfillset(sigset_t *set);

- 功能:将信号集中的所有的标志位置为1

- 参数:set,传出参数,需要操作的信号集

- 返回值:成功返回0, 失败返回-1

int sigaddset(sigset_t *set, int signum);

- 功能:设置信号集中的某一个信号对应的标志位为1,表示阻塞这个信号

- 参数:

- set:传出参数,需要操作的信号集

- signum:需要设置阻塞的那个信号

- 返回值:成功返回0, 失败返回-1

int sigdelset(sigset_t *set, int signum);

- 功能:设置信号集中的某一个信号对应的标志位为0,表示不阻塞这个信号

- 参数:

- set:传出参数,需要操作的信号集

- signum:需要设置不阻塞的那个信号

- 返回值:成功返回0, 失败返回-1

int sigismember(const sigset_t *set, int signum);

- 功能:判断某个信号是否阻塞

- 参数:

- set:需要操作的信号集

- signum:需要判断的那个信号

- 返回值:

1 : signum被阻塞

0 : signum不阻塞

-1 : 失败

cpp 复制代码
/*
    以下信号集相关的函数都是对自定义的信号集进行操作。

    int sigemptyset(sigset_t *set);
        - 功能:清空信号集中的数据,将信号集中的所有的标志位置为0
        - 参数:set,传出参数,需要操作的信号集
        - 返回值:成功返回0, 失败返回-1

    int sigfillset(sigset_t *set);
        - 功能:将信号集中的所有的标志位置为1
        - 参数:set,传出参数,需要操作的信号集
        - 返回值:成功返回0, 失败返回-1

    int sigaddset(sigset_t *set, int signum);
        - 功能:设置信号集中的某一个信号对应的标志位为1,表示阻塞这个信号
        - 参数:
            - set:传出参数,需要操作的信号集
            - signum:需要设置阻塞的那个信号
        - 返回值:成功返回0, 失败返回-1

    int sigdelset(sigset_t *set, int signum);
        - 功能:设置信号集中的某一个信号对应的标志位为0,表示不阻塞这个信号
        - 参数:
            - set:传出参数,需要操作的信号集
            - signum:需要设置不阻塞的那个信号
        - 返回值:成功返回0, 失败返回-1

    int sigismember(const sigset_t *set, int signum);
        - 功能:判断某个信号是否阻塞
        - 参数:
            - set:需要操作的信号集
            - signum:需要判断的那个信号
        - 返回值:
            1 : signum被阻塞
            0 : signum不阻塞
            -1 : 失败

*/
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>

int main(){
    // 创建一个信号集
    sigset_t set;

    //清空信号集的内容
    sigemptyset(&set);
    
    //判断 SIGINT 是否在信号集 set里
    int ret = sigismember(&set,SIGINT);
    if(ret == 0){
        printf("SIGINT 不阻塞\n");
    }else if(ret == 1){
        printf("SIGINT 阻塞\n");
    }else if(ret == -1){
        perror("SIGINT");
        exit(0);
    }

    //添加几个信号到信号集中
    sigaddset(&set,SIGINT);
    sigaddset(&set,SIGQUIT);

    //判断SIGINT是否在信号集中
    ret = sigismember(&set,SIGINT);
    if(ret == 0){
        printf("SIGINT 不阻塞\n");
    }else if(ret == 1){
        printf("SIGINT 阻塞\n");
    }else if(ret == -1){
        perror("SIGINT");
        exit(0);
    }

    //判断SIGQUIT是否在信号集中
    ret = sigismember(&set,SIGQUIT);
    if(ret == 0){
        printf("SIGQUIT 不阻塞\n");
    }else if(ret == 1){
        printf("SIGQUIT 阻塞\n");
    }else if(ret == -1){
        perror("SIGQUIT");
        exit(0);
    }

    //从信号集中删除一个信号
    sigdelset(&set,SIGQUIT);

    //判断SIGQUIT是否在信号集中
    ret = sigismember(&set,SIGQUIT);
    if(ret == 0){
        printf("SIGQUIT 不阻塞\n");
    }else if(ret == 1){
        printf("SIGQUIT 阻塞\n");
    }else if(ret == -1){
        perror("SIGQUIT");
        exit(0);
    }

    return 0;
}
相关推荐
Java.熵减码农6 小时前
解决Linux修改环境变量后导致登录循环进不去系统的问题
linux·运维·服务器
计算机毕设VX:Fegn08956 小时前
计算机毕业设计|基于springboot + vue医院设备管理系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
明天好,会的6 小时前
分形生成实验(五):人机协同破局--30万token揭示Actix-web状态管理的微妙边界
运维·服务器·前端
Mr__Miss6 小时前
保持redis和数据库一致性(双写一致性)
数据库·redis·spring
天骄t6 小时前
嵌入式系统与51单片机核心原理
linux·单片机·51单片机
阿部多瑞 ABU7 小时前
`chenmo` —— 可编程元叙事引擎 V2.3+
linux·人工智能·python·ai写作
Knight_AL7 小时前
Spring 事务传播行为 + 事务失效原因 + 传播行为为什么不用其他模式
数据库·sql·spring
倔强的石头_7 小时前
时序数据时代的“存储与分析困局”解析及金仓解决方案
数据库
计算机毕设VX:Fegn08957 小时前
计算机毕业设计|基于springboot + vue小型房屋租赁系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
徐同保8 小时前
nginx转发,指向一个可以正常访问的网站
linux·服务器·nginx