Linux初识epoll

简介: 简答epoll介绍


本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。


概述

  • eventpoll
  • 高效:

    • 底层为红黑树
    • 使用回调机制而不是线性扫描,处理效率不会随着集合的变大而下降
    • 使用共享内存(内核与用户区之间),避免了频繁拷贝
  • 没有最大文件描述符限制:将用户所关心的文件描述符上的事件放在内核的一个事件表中,同时epoll需要一个额外的文件描述符来唯一标识内核中的这个事件表
  • 线程安全,无需进行共享资源管理

函数

#include <sys/epoll.h>
// 创建epoll实例
int epoll_create(int size);
  • size参数无实际意义
  • 返回内核中epoll事件表的文件描述符
#include <sys/epoll.h>
// 管理epoll红黑树(添加, 修改, 删除)
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
  • fd:要操作的文件描述符
  • op:option

    • EPOLL_CTL_ADD: 向事件表中注册fd上的事件
    • EPOLL_CTL_MOD: 修改fd上的注册事件
    • EPOLL_CTL_DEL: 删除fd上的注册事件
  • event:
struct epoll_event
{
  __uint32_t events;  // epoll事件
  epoll_data_t data;  // 用户数据
}
  • events:描述用户事件

    • EPOLLIN:读事件
    • EPOLLET:边缘触发模式
    • EPOLLONESHTO:在注册了此事件的文件描述符上最多触发一个可读/可写/异常事件(进一步减少可读,可写和异常等事件被触发的次数)
  • data:用于存储用户数据:其为联合体,有四种方式描述用户,最常用的为fd
typedef union epoll_data
{
  void *ptr;
  int fd;
  uint32_t u32;
  uint43_t u64;
}epoll_data_t;
#include <sys/epoll.h>
// 检测epoll树中是否有就绪的文件描述符
int epoll_wait(int epfd, struct epoll_event *event, int maxevents, int timeout);
  • 主要接口
  • epoll_wait如果检测到事件,就将所有就绪的事件从内核表中复制到第二个参数event指向的数组中(这个数组只用于输出就绪事件),极大提高了效率
  • 处理epoll返回的就绪文件描述符:
int ret = epoll_wait(epfd, events, MAX_EVENT_NUMBER, -1);
// 仅遍历ret个文件描述符
for(int i = 0; i < ret; i++) {
  int sockfd = events[i].data.fd;
}

工作模式

水平模式

  • level trigger(LT)
  • 默认工作方式,支持blockno-block socket
  • 通知次数多,易于编写,但效率低
  • 读事件:

    • 文件描述符对应的读缓冲区还有数据,读事件就会被触发,epoll_wait()解除阻塞
    • 读缓冲区一次没有读完,读事件一直触发
    • 读数据是被被动的,必须通过读事件才知道有数据到达了,因此对于读事件的检测是必须
  • 写事件:

    • 如果文件描述符对应的写缓冲区可写,写事件就会被触发,epoll_wait()解除阻塞
    • 如果写缓冲区没有写满,写事件一直触发
    • 写数据是主动的,并且写缓冲区一般都是可写的(未满),所以对写事件的检测不是必须

边缘触发

  • edge trigger(ET)
  • 高速工作模式,只支持no-block socket
  • 通知次数少(只有新事件才会通知,降低了同一事件被重复触发的次数),编写较难,但效率高
  • 读事件:

    • 当读缓冲区有新的数据进入,读事件触发一次,没有新数据不会触发事件
    • 如果数据没有被全部取走,并且没有新数据进入,读时间不会再次出发,只通知一次
    • 如果数据被全部取走或只取走一部分,此时有新数据进入,读事件触发,并且只通知一次
  • 写事件:

    • 当写缓冲区可写,写事件只触发一次
    • 写缓冲区从不满到被写满,期间写事件只触发一次
    • 写缓冲区从满到不满,状态变为可写,写事件只会被触发一次

refer

目录
相关文章
|
4月前
|
网络协议 Linux C++
Linux C/C++ 开发(学习笔记十二 ):TCP服务器(并发网络编程io多路复用epoll)
Linux C/C++ 开发(学习笔记十二 ):TCP服务器(并发网络编程io多路复用epoll)
59 0
|
26天前
|
网络协议 Linux Python
Python网络编程基础(Socket编程)epoll在Linux下的使用
【4月更文挑战第12天】在上一节中,我们介绍了使用`select`模块来实现非阻塞IO的方法。然而,`select`模块在处理大量并发连接时可能会存在性能问题。在Linux系统中,`epoll`机制提供了更高效的IO多路复用方式,能够更好地处理大量并发连接。
|
7月前
|
监控 网络协议 Java
I/O多路复用【Linux/网络】(C++实现select、poll和epoll服务器)(上)
I/O多路复用【Linux/网络】(C++实现select、poll和epoll服务器)
111 0
|
2月前
|
监控 网络协议 Linux
Linux I/O多路复用深入解析:从select到epoll的演进之路
Linux I/O多路复用深入解析:从select到epoll的演进之路
73 0
|
2月前
|
NoSQL Java Linux
【Linux IO多路复用 】 Linux 网络编程 认知负荷与Epoll:高性能I-O多路复用的实现与优化
【Linux IO多路复用 】 Linux 网络编程 认知负荷与Epoll:高性能I-O多路复用的实现与优化
67 0
|
7月前
|
存储 监控 网络协议
I/O多路复用【Linux/网络】(C++实现select、poll和epoll服务器)(下)
I/O多路复用【Linux/网络】(C++实现select、poll和epoll服务器)
129 0
|
4月前
|
网络协议 Linux
【Linux C TCP服务器端-epoll案例】
本文主要介绍了linux下Select的TCP通信流程,实现了客户端和服务器的通信,主要实现了消息的回发,即服务器将消息原封不动的回发给客户端。
32 0
|
4月前
|
网络协议 Linux API
Linux C TCP编程(socket,select/poll/epoll)
本文主要介绍了linux下标准的TCP通信流程,实现了客户端和服务器的通信,主要实现了消息的回发,即服务器将消息原封不动的回发给客户端。如果对接口不熟悉可以参考socket api介绍或者参考其他博客。
27 0
|
5月前
|
Linux
Linux网络编程(epoll的ET模式和LT模式)
Linux网络编程(epoll的ET模式和LT模式)
49 0
|
5月前
|
存储 Linux
Linux网络编程(epoll函数的使用)
Linux网络编程(epoll函数的使用)
49 0