C++ epoll

作者: 周果冻 | 来源:发表于2018-04-13 12:42 被阅读268次

    服务端代码

    int listenfd = socket(AF_INET, SOCK_STREAM, 0);  //生成 socket 文件描述符
    SetSocket(listenfd);                             //设置 socket 为非阻塞
    
    struct epoll_event epev;  //记录套接字相关信息
    epev.events = EPOLLIN;    //监视有数据可读事件
    epev.data.fd = listenfd;  //文件描述符数据,其实这里可以放任何数据。
    
    int epollfd = epoll_create(1024);
    //创建 epoll 文件描述符,出错返回-1
    
    epoll_ctl(epollfd, EPOLL_CTL_ADD, listenfd, &epev);
    //加入监听列表,当 listenfd 上有对应事件产生时, epoll_wait 会将 epoll_event 填充到 events_in 数组里
    //出错返回 -1
    
    struct epoll_event events_in[16];
    while(true)
    {
        int event_count = epoll_wait(epollfd, events_in, 16, -1);
        //等待事件, epoll_wait 会将事件填充至 events_in 内
        //返回 获得的事件数量,若超时且没有任何事件返回 0 ,出错返回 -1 。 timeout设置为-1表示无限等待。
    
        for (int i = 0; i < event_count; i++)      //遍历所有事件
        {
            if (events_in[i].data.fd == listenfd)  //新连接请求
            {
                int new_fd = accept(listenfd, NULL, NULL);
                epev.events = EPOLLIN;  //参见 man 7 epoll 如果要使用 Edge Trigger 还需将 new_fd 设为非阻塞
                epev.data.fd = new_fd;
                epoll_ctl(epollfd, EPOLL_CTL_ADD, new_fd, &epev);  //将新连接加入监视列表
            }
            else if (events_in[i].events & EPOLLIN)  //有数据可读
            {
                int new_fd = events_in[i].data.fd;
                // ... handle(new_fd);
                epoll_ctl(epollfd, EPOLL_CTL_DEL, new_fd, NULL); //不再监听fd,最后一个参数被忽略
                close(new_fd);
            }
        }
    }
    

    epoll_create( )

    int epoll_create(int size);
    /* 生成 epoll 文件描述符,用来存放 socket fd 上是否发生以及发生了什么事件。
    size: 返回的 epoll fd 上能关注的最大 socket fd 数。
    */
    

    epoll_ctl( )

    int epoll_ctl(int epfd, int op, int fd, struct epoll_event* event);
    /* 控制 epoll 文件描述符上的事件:注册、修改、删除。
    epfd: epoll_create() 创建的 epoll 文件描述符。
    op: epoll 监听事件,
      EPOLL_CTL_ADD 增加一个新的 epoll 事件。
      EPOLL_CTL_DEL 减少一个 epoll 事件,
      EPOLL_CTL_MOD 改变一个事件的监听方式。
    fd: socket() 返回值
    event: struct epoll_event;
    */
    

    struct epoll_event

    // 结构体 epoll_event 被用于注册事件和回传所发生待处理的事件,定义如下:
    typedef union epoll_data
    {
        void* ptr;
        int fd;
        __uint32_t u32;
        __uint64_t u64;
    }
    epoll_data_t;  /* 保存触发事件的某个文件描述符相关的数据 */
    
    struct epoll_event
    {
        __uint32_t events;  /* epoll event */
        epoll_data_t data;  /* User data variable */
    };
    /* epoll_event.events:
      EPOLLIN  表示对应的文件描述符可以读
      EPOLLOUT 表示对应的文件描述符可以写
      EPOLLPRI 表示对应的文件描述符有紧急的数据可读
      EPOLLERR 表示对应的文件描述符发生错误
      EPOLLHUP 表示对应的文件描述符被挂断
      EPOLLET  表示对应的文件描述符有事件发生
    */
    

    epoll_wait()

    int epoll_wait(int epfd, struct epoll_event* events,int maxevents, int timeout);
    /* 等待I/O事件的发生
    epfd: epoll_create() 生成的 epoll 文件描述符;
    epoll_event: 用于回传代处理事件的数组;
    maxevents: 每次能处理的事件数;
    timeout: 等待I/O事件发生的超时值;返回发生事件数。
    

    相关文章

      网友评论

          本文标题:C++ epoll

          本文链接:https://www.haomeiwen.com/subject/kfdckftx.html