epoll基本使用

epoll简介

epoll是Linux中的一种使用IO多路复用机制,即使用单个线程就可以监听多个文件描述符的读写事件,避免开启大量线程或者执行大量轮询的造成的不必要的性能开销。类似的机制还有select和poll,关于其区别参考。

高性能IO模型分析-浅析Select、Poll、Epoll机制(三) - 知乎 (zhihu.com)

头文件和API

#include <sys/epoll.h>
// 创建epoll,通常使用epoll_create即可,epoll_create1可以设置一些额外的配置
int epoll_create(int size);
int epoll_create1(int flags);

// 向epoll中添加、删除、修改fd
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);

// 等待fd的事件
int epoll_wait(int epfd, struct epoll_event *events,
               int maxevents, int timeout);
// epoll_pwait相比epoll_wait多了屏蔽指定信号的功能,即通过指定sigmask可以在
// epoll_pwait函数中修改信号屏蔽设置,epoll_pwait返回后恢复原始信号屏蔽设置
int epoll_pwait(int epfd, struct epoll_event *events,
                int maxevents, int timeout,
                const sigset_t *sigmask);
// epoll_pwait2和epoll_wait区别在于timeout可以指定的精度不同,epoll_pwait2
// 可以指定纳秒级别的精度(实际内核调度精度达不到),上边函数的timeout精度为毫秒
int epoll_pwait2(int epfd, struct epoll_event *events,
                 int maxevents, const struct timespec *timeout,
                 const sigset_t *sigmask);
#include <unistd.h>
// 关闭并释放epoll
int close(int fd);

从Linux2.6.8开始,epoll_create中的size实际上没有作用了

epoll_create(2) - Linux manual page (man7.org)

epoll_ctl(2) - Linux manual page (man7.org)

关键数据结构

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_ctl设置需要监听的fd的事件事件属性以及附带的额外数据,第二是在epoll_wait获取触发对应事件以及额外数据

events是需要监听的文件描述符事件以及事件属性,其为一个bitmask,可以是如下内容通过或运算(|)得到的值(仅包含常用部分,全部内容见参考文献):

events说明
EPOLLIN文件描述符中有数据可以读取(事件)
EPOLLOUT文件描述符可以写入数据(事件)
EPOLLRDHUP (since Linux 2.6.17)对端关闭连接(事件)
EPOLLET使用边沿触发(即仅在事件发生(例如有数据可读)时epoll_wait返回一次,即时数据没有读取完,第二次调用epoll_wait时仍然阻塞)。默认状态下epoll使用水平触发,即如果有文件描述符的数据没有读取完则每次调用epoll_wait均不会阻塞。

data通常用于设置对应的文件描述符(fd)。在epoll_ctl中设置的epoll_data可以在epoll_wait中获取到,因此通常将epoll_data设置为文件描述符以便于在epoll_wait返回后操作对应文件。

epoll_ctl中op的合法值

op描述
EPOLL_CTL_ADD将一个fd加入到epoll中
EPOLL_CTL_MOD使用新的epoll_event来修改已经加入到epoll中的fd的设置
EPOLL_CTL_DEL从epoll中删除一个fd,events可以设置为NULL

参考

epoll_ctl(2) - Linux manual page (man7.org)

epoll(7) - Linux manual page (man7.org)

基本使用

#define MAX_FD 3
#define LOG_ERROR(fmt, ...) printf("[ERROR]" fmt "\n", ##__VA_ARGS__)
void demo() {
    int ret;
    int in_fd;
    int epfd = -1;
    // 需要监听的socket
    int in_socks[MAX_FD] = {0, 1, 2};
    // 在添加socket到epoll时使用的的epoll_event结构
    struct epoll_event set_event;
    // epoll_event数组用于接收准备就绪的fd
    struct epoll_event ep_events[MAX_FD];
    
	// 创建epoll(从Linux2.6.8开始,epoll_create的参数已经没有作用,设置为任何大于等于0的数即可)
    epfd = epoll_create(MAX_FD);
    if (epfd < 0) {
        LOG_ERROR("Fail to create epoll, %s", strerror(epfd));
        goto out;
    }
	// 将需要监听的socket加入到epoll中
    for (i=0; i<MAX_FD; i++) {
        // 设置事件以及属性
        set_event.events = EPOLLIN;
        // 设置额外数据,通过额外数据可以直接是fd或者能够获取到fd的索引,这里直接设置为fd
       	set_event.data.fd = in_socks[i];
        // 将socket加入对应的epoll
        ret = epoll_ctl(epfd, EPOLL_CTL_ADD, in_socks[i], &set_event);
        if (ret < 0) {
            LOG_ERROR("Fail to add fd to epoll, %s", strerror(errno));
            goto out;
        }
    }

    while (1) {
        // 等待fd中的数据就绪,timeout设置为-1表示一直阻塞等待
        ready_fd_num = epoll_wait(epfd, ep_events, MAX_FD, -1);
        if (ready_fd_num < 0) {
            LOG_ERROR("Fail to wait for data, err: %s", strerror(errno));
            goto out;
        }
		// 处理就绪的fd
        for(int i=0; i<ready_fd_num; i++) {
            in_fd = ep_events[i].data.fd;
            // 处理数据
        }
    }
out:
    if (epfd != -1) {
        close(epfd);
    }
    return;
}

附录

下边是部分Linux中的API和其头文件

#include <errno.h>
// errno全局变量
#include <string.h>
char *strerror(int errnum);
#include <stdlib.h>

void exit(int status);
  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: `epoll` 函数是 Linux 系统中一个用于处理大量并发连接的 I/O 多路复用机制。它通过维护一个文件描述符集合来监测多个描述符的状态,以便在发生 I/O 事件时快速通知程序。 使用 `epoll` 的基本流程如下: 1. 创建 `epoll` 句柄:使用 `epoll_create` 或 `epoll_create1` 函数创建一个 `epoll` 句柄。 2. 注册文件描述符:使用 `epoll_ctl` 函数向 `epoll` 句柄中添加需要监测的文件描述符,并为每个文件描述符设置监测事件。 3. 进行等待:使用 `epoll_wait` 函数阻塞等待,直到有一个或多个文件描述符准备就绪。 4. 处理事件:当有文件描述符就绪时,处理相应的 I/O 事件。 5. 重复步骤 3 和 4,不断监测文件描述符的状态,处理 I/O 事件。 常见的使用方式是:创建一个线程,在线程中调用 `epoll_wait` 函数,当有文件描述符就绪时通过回调函数处理相应的 I/O 事件。 ### 回答2: epoll函数是Linux系统中一种高效的I/O事件通知机制,用于管理大量的文件描述符。其使用方式如下: 1. 创建一个epoll句柄: int epoll_create(int size); 创建一个epoll实例,并返回一个文件描述符,size表示期望监听的文件描述符数量,通常可以设置为任意正整数。 2. 注册文件描述符和事件: 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. 开始监听文件描述符事件: int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout); epfd为epoll实例的文件描述符,events为存储事件的数组,maxevents表示最大监听的事件数量,timeout表示等待时间(-1表示一直等待,0表示立即返回,>0表示超时时间)。 4. 对返回的事件进行处理: 在epoll_wait函数返回后,可以遍历events数组,根据每个事件的文件描述符和事件类型进行处理。 5. 关闭epoll实例: close(epfd); 使用epoll实例后,需要调用close函数关闭,释放相关资源。 以上就是epoll函数的基本使用流程。通过epoll可以高效地监听大量的文件描述符事件,减少系统资源的消耗。在实际开发中,可以根据需要设置不同的事件类型和回调函数,实现具体的业务逻辑。 ### 回答3: epoll函数是Linux中用于处理I/O事件的一种高效机制。它可以监视一组文件描述符,并在其中的任意一个文件描述符上发生事件时进行相应的处理。 使用epoll函数的基本步骤如下: 1. 调用epoll_create函数创建一个epoll的句柄,该句柄被用于后续的相关操作。 2. 使用epoll_ctl函数向epoll句柄中注册需要监视的文件描述符和事件。通过该函数可以实现添加、修改和删除文件描述符以及相应事件的功能。 3. 使用epoll_wait函数等待事件的发生。epoll_wait会一直阻塞,直到有文件描述符上的事件发生。一旦有事件发生,epoll_wait会返回所发生事件的文件描述符和相应的事件类型。 4. 根据返回的事件类型,进行相应的处理。 epoll函数有三个基本的系统调用: 1. epoll_create函数用来创建一个epoll实例,返回一个epoll句柄。 2. epoll_ctl函数用于操作epoll实例,可以实现添加、修改和删除文件描述符以及相应事件的功能。 3. epoll_wait函数用于等待事件的发生,一旦事件发生则返回相应的文件描述符和事件类型。 epoll函数的使用优点包括: - 支持大量的连接,可以监视数万个文件描述符。 - 存储监视文件描述符的数据结构(epoll实例)可以重复利用,避免了每次都需要重新设置的问题。 - 使用epoll_wait函数进行等待事件的发生,避免了轮询的方式,提高了效率。 总之,epoll函数是Linux中用于处理I/O事件的一种高效机制,可以通过创建、操作epoll实例和等待事件的发生来实现对文件描述符的监视和相应事件的处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值