epoll初识
epoll是为了处理大量的句柄而作了改进的poll;
它几乎具备了之前所说的一切优点,被公认为Linux2.6下性能最好的多路I/O就绪通知方法
epoll相关系统调用
epoll_create -> 创建epoll模型(即内核中的软件资源)
// 创建一个epoll句柄
int epoll_create(int size);
-
自Linux2.6.8以后,size的参数是被忽略的,不过为了可移植性,一般还是会填128,512等值
-
返回值是一个文件描述符,用完之后需要close掉
epoll_ctl -> epoll的事件注册函数
int epoll_ctl(int epfd, int op, int fd, struct epoll_event* event);
-
它不同于select()是在监听事件时告诉内核要监听什么样的事件,而是利用这个函数先注册要监听的事件
参数:
-
epfd: 是epll_create()返回的epoll句柄
-
op: 表示动作,用一下三个宏来表示
-
EPOLL_CTL_ADD: 注册新的fd到epfd中
-
EPOLL_CTL_MOD: 修改已经注册的fd的监听事件
-
EPOLL_CTL_DEL: 从epfd中删除一个fd
-
-
fd: 需要监听的文件描述符
-
event: 告诉内核需要监听什么事件,结构体内容如下:
-
events可以是以下的宏的集合(这里列出部分值)
|
宏值 |
描述 |
EPOLLIN |
0x001 |
表示对应文件描述符可以读(包括对端SOCKET正常关闭) |
EPOLLOUT |
0x004 |
表示对应的文件描述符可以写 |
EPOLLPRI |
0x002 |
表示对应的文件描述符有紧急的数据可读(这里应该表示有带外数据到来) |
EPOLLERR |
0x008 |
表示对应文件描述符发生错误 |
EPOLLHUP |
0x010 |
表示对应的文件描述符被挂断 |
-
epoll_data结构体保存的是用户自定义的数据,一般我们使用的是里面的 fd 字段,保存文件描述符
epoll_wait -> 收集在epoll监控事件中已经发送的事件
int epoll_waot(int epfd, struct epoll_event* events, int maxevent, int timeout);
参数:
-
epfd: epoll_create返回的句柄
-
events: 就是上面介绍的结构体,本质上是一个结构体数组,epoll会将发生的事件赋值到events数组中(events不可以是空指针,内核只负责把数据复制到这个events数组中,不会去帮助我们在用户态分配内存)
-
maxevent: 是上面结构体数据的长度,maxevents的值不可以大于epoll_create()的size
-
timeout: 超时时间,和select(),poll()的意义相同
返回值:
- 成功: 返回对应I/O上已经准备好的文件描述符个数
- 失败: 返回值小于0
-
超时: 返回0
epoll工作原理
构造e