epoll详解【epoll_create】【epoll_ctl】【epoll_wait】

0 - 前言

参考:epoll机制:epoll_create、epoll_ctl、epoll_wait、close

1 - epoll接口

epoll_create

#include <sys / epoll.h>
 
nfd = epoll_creat(max_size);

创建一个内核事件表 (也叫epoll句柄),nfd为epoll句柄(不了解句柄看下一段),参数max_size标识这个监听的数目最大有多大,从Linux 2.6.8开始,max_size参数将被忽略,但必须大于零。每个内核事件表占据一个fd。这个函数其实是在内核申请一块空间,用来存放你想关注的socket fd上是否发生以及发生了什么事件。max_size就是你在这个epoll fd上能关注的最大socket fd数。

返回值:成功时,这些系统调用将返回非负文件描述符。如果出错,则返回-1,并且将errno设置为指示错误。

句柄是一个标识符,是拿来标识对象或者项目的,它就象我们的姓名一样,每个人都会有一个

参考:epoll_create详解

epoll_ctl

#include <sys / epoll.h>
 
int epoll_ctl(int epfd,int op,int fd,struct epoll_event * event);

该系统调用对文件描述符epfd引用的epoll实例执行控制操作。它要求操作op对目标文件描述符fd执行。

函数参数:

  • epfd:由epoll_create产生的epoll句柄
  • op:要进行的的操作,比如EPOLL_CTL_ADD(注册新的fd到epfd中)、EPOLL_CTL_MOD( 修改已经注册的fd的监听事件)、EPOLL_CTL_DEL(从epfd中删除一个fd)
  • fd:需要监听的fd
  • event:指向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 events(也就是事件) */
epoll_data_t data; /* User data variable */

epoll_event中的成员events可以是:

  • EPOLLIN:表示对应的文件描述符上有可读数据
  • EPOLLOUT:表示对应的文件描述符上可以写数据
  • EPOLLPRI:表示对应的文件描述符有紧急的数据可读
  • EPOLLERR:表示对应的文件描述符发生错误
  • EPOLLHUP:表示对应的文件描述符被挂断
  • EPOLLET: 将EPOLL设为边缘触发(Edge Triggered)模式,这是相对于水平触发(Level Triggered)来说的
  • EPOLLONESHOT:只监听一次事件,当监听完这次事件之后,如果还需要继续监听这个socket的话,需要再次把这个socket加入到epoll队列里

返回值:成功时,epoll_ctl()返回零。发生错误时,epoll_ctl()返回-1并正确设置了errno

参考:epoll_ctl详解

epoll_wait

#include <sys / epoll.h>
 
int epoll_wait(int epfd,struct epoll_event * events, int maxevents,int timeout);

等待文件描述符epfd引用的epoll实例上的I/O事件

函数参数:

  • epfd:由epoll_create产生的epoll句柄
  • events:从内核得到事件的集合
  • maxevents:内核这个events有多大(数组成员的个数),这个maxevents的值不能大于创建epoll_create()时的size
  • timeout:超时时间(毫秒,0会立即返回,-1将永久阻塞),当超过timeout还没有事件触发时,就超时

返回值:需要处理的事件数目,若返回0表示已超时。

epoll_wait运行的原理:等侍注册在epfd上的socket fd的事件的发生,如果发生则将发生的sokct fd和事件类型放入到events数组中。并且,将注册在epfd上的socket fd的事件类型给清空。所以如果下一个循环你还要关注这个socket fd的话,则需要用epoll_ctl(epfd,EPOLL_CTL_MOD,listenfd,&ev)来重新设置socket fd的事件类型。这时不用EPOLL_CTL_ADD,因为socket fd并未清空,只是事件类型清空。这一步非常重要。

参考:epoll_wait详解

  • 0
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
这三个函数都是与 Linux 系统中的 epoll I/O 多路复用机制相关的函数。 1. `epoll_create` 函数用于创建一个 epoll 文件描述符,返回值即为该文件描述符。该函数的原型为: ```c int epoll_create(int size); ``` 其中参数 `size` 表示 epoll 实例中能够关注的最大文件描述符数目,一般可以设置为任意大于 0 的数。 2. `epoll_ctl` 函数用于控制 epoll 实例中的文件描述符,可以用于添加、修改和删除文件描述符的关注事件。该函数的原型为: ```c int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event); ``` 其中 `epfd` 参数表示 epoll 文件描述符,`op` 参数表示操作类型,可选的取值有 `EPOLL_CTL_ADD`、`EPOLL_CTL_MOD` 和 `EPOLL_CTL_DEL`,分别表示添加、修改和删除操作。`fd` 参数表示需要被添加、修改或删除的文件描述符,`event` 参数表示需要关注的事件,包括事件类型和事件数据等信息。 3. `epoll_wait` 函数用于等待 epoll 实例中的文件描述符上发生关注的事件,该函数会阻塞进程直到有事件发生。该函数的原型为: ```c int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout); ``` 其中 `epfd` 参数表示 epoll 文件描述符,`events` 参数表示事件数组,用于存储返回的事件信息,`maxevents` 参数表示最大返回的事件数目,`timeout` 参数表示超时时间,单位为毫秒,如果设置为 -1 则表示永久阻塞直到有事件发生。 总的来说,epoll I/O 多路复用机制通过 `epoll_create` 函数创建一个 epoll 实例,并通过 `epoll_ctl` 函数控制实例中的文件描述符。然后通过 `epoll_wait` 函数等待文件描述符上发生关注的事件,从而实现高效的 I/O 多路复用。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值