Reactor 模式结合 epoll

一、核心概念

  1. Reactor 模式

    • 事件驱动架构
    • 核心组件:
      • Event Demultiplexer(epoll)
      • Event Handler(事件处理器)
      • Reactor(事件循环)
  2. epoll 机制

    • Linux 高效 I/O 多路复用
    • 三种系统调用:
      • epoll_create():创建实例
      • epoll_ctl():注册事件
      • epoll_wait():等待事件

二、实现架构

复制代码
+------------------+
|    Event Loop    |
+--------+---------+
         | 通过 epoll_wait 获取事件
+--------v---------+
|  Event Demux     |
| (epoll instance) |
+--------+---------+
         | 分发到处理器
+--------v---------+
|  Event Handlers  |
| (accept/read/write) 
+------------------+

三、关键实现代码

1. 创建 epoll 实例
复制代码
int epoll_fd = epoll_create1(0);
if (epoll_fd == -1) {
   
    perror("epoll_create1");
    exit(EXIT_FAILURE);
}
2. 注册事件
复制代码
struct epoll_event ev;
ev.events = EPOLLIN | EPOLLET;  // 边缘触发模式
ev.data.fd = sockfd;

if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, sockfd, &ev) == -1) {
   
    perror("epoll_ctl: listen_sock");
    exit(EXIT_FAILURE);
}
3. 事件循环
复制代码
#define MAX_EVENTS 64
struct epoll_event events[MAX_EVENTS];

while (1) {
   
    int nfds = epoll_wait(epoll_fd, events, MAX_EVENTS, -1);
    if (nfds == -1) {
   
        perror("epoll_wait");
        exit(EXIT_FAILURE);
    }

    for (int i = 0; i < nfds; ++i) {
   
        if (events[i].data.fd == listen_fd) {
   
            handle_accept(listen_fd);
        } else {
   
            if (events[i].events & EPOLLIN) {
   
                handle_read(events[i].data.fd);
            }
            if (events[i].events & EPOLLOUT) {
   
                handle_write(events[i].data.fd);
            }
        }
    }
}

四、关键处理函数示例

1. 接受连接
复制代码
void handle_accept(int listen_fd) {
   
    struct sockaddr_in client_addr;
    socklen_t addrlen = sizeof(client_addr);

    int conn_fd;
    while ((conn_fd = accept(listen_fd, 
                           (struct sockaddr*)&client_addr,
                           &addrlen)) > 0) {
   
        set_nonblocking(conn_fd);

        struct epoll_event ev;
        ev.events = EPOLLIN | EPOLLET;
        ev.data.fd = conn_fd;
        epoll_ctl(epoll_fd, EPOLL_CTL_ADD, conn_fd, &ev);
    }

    if (errno != EAGAIN && errno != EWOULDBLOCK) {
   
        perror("accept error");
    }
}
2. 读取数据
复制代码
void handle_read(int fd) {
   
    char buf[1024];
    ssize_t n;

    while ((n = read(fd, buf, sizeof(buf))) > 0) {
   
        // 处理数据逻辑
        buf[n] = '\0';
        printf("Received: %s\n", buf);

        // 修改为监听写事件
        struct epoll_event ev;
        ev.events = EPOLLOUT | EPOLLET;
        ev.data.fd = fd;
        epoll_ctl(epoll_fd, EPOLL_CTL_MOD, fd, &ev);
    }

    if (n == 0) {
     // 连接关闭
        close(fd);
    } else if (errno != EAGAIN) {
   
        perror("read error");
        close(fd);
    }
}

五、重要注意事项

  1. 非阻塞 I/O

    复制代码
    void set_nonblocking(int fd) {
         
        int flags = fcntl(fd, F_GETFL, 0);
        fcntl(fd, F_SETFL, flags | O_NONBLOCK);
    }
  2. 触发模式选择

    • 水平触发(LT,默认):数据未处理完会持续通知
    • 边缘触发(ET):只在状态变化时通知,必须搭配非阻塞 I/O
  3. 资源管理

    • 及时关闭不需要的文件描述符
    • 合理设置 epoll 事件类型
    • 处理 EAGAIN/EWOULDBLOCK 错误

六、完整示例(TCP Echo Server)

复制代码
#include <sys/epoll.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define PORT 8080
#define MAX_EVENTS 64

void set_nonblocking(int fd) {
   
    int flags = fcntl(fd, F_GETFL, 0);
    fcntl(fd, F_SETFL, flags | O_NONBLOCK);
}

int main() {
   
    int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
    struct sockaddr_in addr = {
   
        .sin_family = AF_INET,
        .sin_port = htons(PORT),
        .sin_addr.s_addr = INADDR_ANY
    };

    bind(listen_fd, (struct sockaddr*)&addr, sizeof(addr));
    listen(listen_fd, SOMAXCONN);

    int epoll_fd = epoll_create1(0);
    struct epoll_event ev;
    ev.events = EPOLLIN | EPOLLET;
    ev.data.fd = listen_fd;
    epoll_ctl(epoll_fd, EPOLL_CTL_ADD, listen_fd, &ev);

    struct epoll_event events[MAX_EVENTS];

    while (1) {
   
        int nfds = epoll_wait(epoll_fd, events, MAX_EVENTS, -1);

        for (int i = 0; i < nfds; ++i) {
   
            if (events[i].data.fd == listen_fd) {
   
                // 处理新连接
                struct sockaddr_in client_addr;
                socklen_t addrlen = sizeof(client_addr);
                int conn_fd;
                while ((conn_fd = accept(listen_fd, 
                        (struct sockaddr*)&client_addr,
                        &addrlen)) > 0) {
   
                    set_nonblocking(conn_fd);
                    struct epoll_event ev_conn;
                    ev_conn.events = EPOLLIN | EPOLLET;
                    ev_conn.data.fd = conn_fd;
                    epoll_ctl(epoll_fd, EPOLL_CTL_ADD, conn_fd, &ev_conn);
                }
            } else {
   
                // 处理数据
                if (events[i].events & EPOLLIN) {
   
                    char buf[1024];
                    ssize_t n;
                    while ((n = read(events[i].data.fd, buf, sizeof(buf))) > 0) {
   
                        write(events[i].data.fd, buf, n); // Echo 回显
                    }

                    if (n == 0 || (n < 0 && errno != EAGAIN)) {
   
                        close(events[i].data.fd);
                    }
                }
            }
        }
    }

    close(epoll_fd);
    close(listen_fd);
    return 0;
}
相关推荐
易连EDI—EasyLink1 小时前
易连EDI–EasyLink实现OCR智能数据采集
网络·人工智能·安全·汽车·ocr·edi
@insist1232 小时前
信息安全工程师考点精讲:身份认证核心原理与分类体系(上篇)
大数据·网络·分类·信息安全工程师·软件水平考试
SmartRadio2 小时前
ESP32-S3 双模式切换实现:兼顾手机_路由器连接与WiFi长距离通信
开发语言·网络·智能手机·esp32·长距离wifi
_.Switch2 小时前
东方财富股票数据JS逆向:secids字段和AES加密实战
开发语言·前端·javascript·网络·爬虫·python·ecmascript
金色光环3 小时前
FreeModbus释放底层的 TCP 监听端口
服务器·网络·tcp/ip
数智化精益手记局3 小时前
拆解物料管理erp系统的核心功能,看物料管理erp系统如何解决库存积压与缺料难题
大数据·网络·人工智能·安全·信息可视化·精益工程
灰子学技术5 小时前
Envoy HTTP 过滤器处理技术文档
网络·网络协议·http
Olivia051405147 小时前
Voohu:音频变压器的屏蔽接地技术对50Hz工频噪声抑制的影响
网络·机器人·信息与通信
byoass7 小时前
智巢AI知识库深度解析:企业文档管理从大海捞针到精准狙击的进化之路
开发语言·网络·人工智能·安全·c#·云计算
zhihuishuxia__7 小时前
Multiplex通讯(多路复用通讯)
网络·图像处理·数码相机·计算机视觉·自动化