【1】epoll相关的系统调用
epoll是Linux特有的I/O复用函数。它在实现和使用上与select、poll有很大差异。
首先,epoll使用一组函数来完成任务,而不是单个函数。
其次,epoll 把用户关心的文件描述符上的事件放在内核里的一个事件表中,从而无须像select和poll那样每次调用都要重复传人文件描述符集或事件集。
但是epoll需要使用一个额外的文件描述符,来唯一标识内核中的这个事件表。
epoll_create()
这个文件描述符使用如下epoll create函数来创建:
#include<sys/epoll.h>
int epoll_create(int size)
其中size参数现在并不起作用,只是给内核一个提示,告诉它事件表需要多大。
返回值
该函数返回一个文件描述符,用作其他所有epoll系统调用的第一个参数,用来指定要访问的内核事件表。
epoll_ctl()
#include <sys/epoll.h>
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
参数epfd是通过epoll_create创建的epoll实例
参数fd是将要被监视的文件描述符
参数op是epoll操作符
其中参数op有如下三个有效值如下
- EPOLL_CTL_ADD ,向epfd注册fd的上的event
- EPOLL_CTL_MOD,修改fd已注册的event
- EPOLL_CTL_DEL,从epfd上删除fd的event
参数event指定事件,他是一个epoll_event结构指针类型如下:
struct epoll_event
{
__uint32_t events; // epoll事件
epoll_data_t data; // 用户数据
};
其中events成员描述事件类型,epoll支持的事件类型和poll基本相同,唯一的区别就是在poll的事件类型前加上E代表epoll,如下:
事件 | 描述 |
---|---|
EPOLLIN | 有数据可读 |
EPOLLRDNORM | 有普通数据可读 |
EPOLLRDBAND | 有优先数据可读 |
EPOLLPRI | 有紧迫数据可读 |
EPOLLOUT | 写数据不会导致阻塞 |
EPOLLWRNORM | 写普通数据不会导致阻塞 |
EPOLLWRBAND | 写优先数据不会导致阻塞 |
EPOLLMSGSIGPOLL | 消息可用 |
data成员用于存储用户数据,其类型为epoll_data_t定义如下:
typedef union epoll_data
{
void *ptr;
int fd;
__uint32_t u32;
__uint64_t u64;
} epoll_data_t;
epoll_data_t是一个联合体.
fd是我们最常用的成员,他指定事件所从属的目标文件描述符
ptr可指定与fd相关的用户数据。
返回值
epoll_ctl()执行成功时返回0,失败时返回-1,并置errno
epoll_wait()
//返回就绪事件描述符的个数
#include<sys/epoll.h>
int epoll_wait(int epfd,struct epoll_event *events,int maxevents,int timeout);
epoll_wait()用于等待事件的就绪,类似于select()的调用。
参数events用来从内核得到事件的集合。
参数maxevents告诉内核这个events有多大,这个maxevents的值不能大于创建epoll_create()时的size。
参数timeout是超时时间(毫秒,0会立即返回,-1将不确定,也有说法说是永久阻塞)
返回值
该函数调用成功时返回需要处理的事件个数,如返回0表示已超时。
【2】代码示例
使用epoll实现多客户端服务器的交互
server.c
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#i