通过与 Jira 对比,让您更全面了解 PingCode

  • 首页
  • 需求与产品管理
  • 项目管理
  • 测试与缺陷管理
  • 知识管理
  • 效能度量
        • 更多产品

          客户为中心的产品管理工具

          专业的软件研发项目管理工具

          简单易用的团队知识库管理

          可量化的研发效能度量工具

          测试用例维护与计划执行

          以团队为中心的协作沟通

          研发工作流自动化工具

          账号认证与安全管理工具

          Why PingCode
          为什么选择 PingCode ?

          6000+企业信赖之选,为研发团队降本增效

        • 行业解决方案
          先进制造(即将上线)
        • 解决方案1
        • 解决方案2
  • Jira替代方案

25人以下免费

目录

signalfd 是什么

信号(signal)本质是 Linux 进程间通信的一种机制,也叫软中断信号。signalfd 是一个跟信号关联的文件描述符,能够以 io 的行为获取到系统信号,属性上来讲 signalfd 也是一个匿名 fd 类型。

一、signalfd 是什么

信号(signal)本质是 Linux 进程间通信的一种机制,也叫软中断信号。signalfd 是一个跟信号关联的文件描述符,能够以 io 的行为获取到系统信号,属性上来讲 signalfd 也是一个匿名 fd 类型:

root@ubuntu:~# ll /proc/15445/fd

lrwx—— 1 root root 64 Aug 24 16:42 3 -> anon_inode:[signalfd]

root@ubuntu:~# cat /proc/15445/fdinfo/3

pos: 0

flags: 02

mnt_id: 11

sigmask: 0000000000000006

从这里可以得到简单的信息:

  1. signal 用的匿名 inode ,signalfd 属于匿名 fd 的一种;
  2. 句柄关联的重要信息就是 sigmask,通过 /proc/${pid}/fdinfo/3 能看到这个值

其实信号是很讲究的,甚至有信号编程一说,Linux 的 signalfd 为信号的处理提供了一种新的方法,统一到文件的 io 模式,契合一切接文件的理念。

系统调用:

#include <sys/signalfd.h>

int signalfd(int fd, const sigset_t *mask, int flags);

该系统调用返回一个整数类型 signalfd,这个句柄跟信号行为绑定,当发生信号的时候,句柄触发可读事件。

名列前茅个参数也可以传入一个有效的信号 fd 的句柄,如果传入的是 -1 ,那么内核会自动创建一个新的 fd 。

完整的代码例子,在 Linux 机器上,通过 man signalfd 就可以获取到。

  1. 信号能够像文件一样 read 出来,这种优雅的信号处理方式得益于 signalfd 的封装;
  2. 信号是挂在在进程 task_struct 结构体上的,信号队列非空的时候 signalfd 句柄可读;
  3. 和 epoll 池的配合同样还是老套路,epoll_ctl 注册的时候调用 .poll 接口挂载 epoll 的 wait entry 到 sighand->signalfd_wqh 之上,信号发送时()唤醒 epoll ;
  4. signalfd 也是一种匿名 fd 类型。

延伸阅读:

二、信号的使用示例

我们通过一段代码实例来看一下信号量的使用吧。

void IntHandler(int signum) {

  std::cout << time(NULL) << ” Got a int signal” << signum << std::endl;

  std::this_thread::sleep_for(5s);

  std::cout << time(NULL) << ” Fininsh int signal” << signum << std::endl;

}

int main(int argc, char* argv[]) {

  signal(SIGINT, IntHandler);

  while(true) {

    std::this_thread::sleep_for(10s);

    std::cout << “.” << std::endl;

  }

  std::cout << std::endl;

  return 0;

}

上面这段代码,我们通过signal(SIGINT, IntHandler);自定义了SIGINT信号量的处理。程序运行起来后,当按下ctrl + c时,IntHandler信号处理函数被触发。

相关文章