epoll监听信号事件-signalfd

signalfd介绍

signalfd:传统的处理信号的方式是注册信号处理函数;由于信号是异步发生的,要解决数据的并发访问,可重入问题。signalfd可以将信号抽象为一个文件描述符,当有信号发生时可以对其read,这样可以将信号的监听放到select、poll、epoll等监听队列中。当有事件触发时,有可读事件发生。
signalfd涉及API:

#include <sys/signalfd.h>
int signalfd(int fd, const sigset_t *mask, int flags);

参数fd:如果是-1则表示新建一个,不是-1并且是一个合法的signalfd表示向其添加新的信号。

参数mask:信号集合。

参数flag:内核版本2.6.27以后支持SFD_NONBLOCK、SFD_CLOEXEC。

成功返回文件描述符,返回的fd支持以下操作:read、select(poll、epoll)、close。

  1. 将感兴趣的信号加入到sigset_t中
  2. 调用signalfd(),把信号集与fd关联起来
  3. 调用sigprocmask(),把信号集添加进去,避免触发这些信号的默认处理方式
  4. 阻塞等待信号的发生并读取
  5. 与传统的处理信号的方式一样,发送多个相同的信号,只处理一次

epoll+signalfd

#include <signal.h>
#include <string>
#include <sys/epoll.h>
#include <sys/signalfd.h>
#include <unistd.h>
#include<iostream>

void signalCallBackFunc(int signalFd) {
  struct signalfd_siginfo fdsiI;
  std::cout << "signalCallBackFunc" << std::endl;
  int s = read(signalFd, &fdsiI, sizeof(struct signalfd_siginfo));
}

int main() {
  int epfd = epoll_create1(0);
  sigset_t sigintMask;

  sigemptyset(&sigintMask);
  sigaddset(&sigintMask, SIGINT);
  sigprocmask(SIG_BLOCK, &sigintMask, NULL);

  int sigintfd;
  sigintfd = signalfd(-1, &sigintMask, 0);

  struct epoll_event sigintEvent;
  sigintEvent.data.fd = sigintfd;
  sigintEvent.events = EPOLLIN;

  epoll_ctl(epfd, EPOLL_CTL_ADD, sigintfd, &sigintEvent);

  struct epoll_event events[1024];
  while (1) {
    int n = epoll_wait(epfd, events, 1024, -1);

    for (int i = 0; i < n; i++) {
      auto returnfd = events[i].data.fd;
      signalCallBackFunc(returnfd);
    }
  }
}
posted @ 2021-09-23 21:37  cheems~  阅读(346)  评论(0编辑  收藏  举报