epoll_ctl的概念和使用案例

epoll_ctl 是 Linux 系统中 I/O 多路复用机制 epoll 的核心函数之一,用于管理 epoll 实例监控的文件描述符(File Descriptor, FD)。它负责向 epoll 实例注册、修改或删除需要监控的 FD 及其事件类型,是实现高性能网络编程(如高并发服务器)的关键工具。


函数原型

#include <sys/epoll.h>

int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
参数说明
参数说明
epfdepoll 实例的文件描述符(由 epoll_create 创建)
op操作类型:EPOLL_CTL_ADD(添加)、EPOLL_CTL_MOD(修改)、EPOLL_CTL_DEL(删除)
fd需要操作的目标文件描述符(如 socket)
event指向 epoll_event 结构体的指针,定义监控的事件类型和用户数据
返回值
  • 成功返回 0,失败返回 -1,错误信息通过 errno 获取。

epoll_event 结构体

struct epoll_event {
    uint32_t     events;   // 监控的事件类型(位掩码形式)
    epoll_data_t data;     // 用户数据(通常保存 FD 或关联的指针)
};

typedef union epoll_data {
    void    *ptr;
    int      fd;
    uint32_t u32;
    uint64_t u64;
} epoll_data_t;
常用事件类型
事件类型说明
EPOLLIN文件描述符可读(例如 socket 接收缓冲区有数据)
EPOLLOUT文件描述符可写(例如 socket 发送缓冲区有空闲)
EPOLLERR发生错误(自动监控,无需显式设置)
EPOLLHUP对端关闭连接或挂起(自动监控)
EPOLLET边缘触发模式(Edge-Triggered),默认为水平触发(Level-Triggered)

使用场景案例:TCP 服务器监控 Socket

以下是一个简化的 TCP 服务器代码片段,展示 epoll_ctl 的典型用法:

1. 创建 epoll 实例
int epfd = epoll_create1(0);  // 创建 epoll 实例
if (epfd == -1) {
    perror("epoll_create1");
    exit(EXIT_FAILURE);
}
2. 注册监听 Socket 到 epoll
int listen_sock = socket(AF_INET, SOCK_STREAM, 0);  // 创建监听 socket
struct sockaddr_in addr = {/* 绑定 IP 和端口 */};
bind(listen_sock, (struct sockaddr*)&addr, sizeof(addr));
listen(listen_sock, SOMAXCONN);  // 开始监听

// 定义 epoll_event 结构体
struct epoll_event ev;
ev.events = EPOLLIN | EPOLLET;  // 监控可读事件,边缘触发模式
ev.data.fd = listen_sock;       // 保存 socket FD 到用户数据

// 将监听 socket 添加到 epoll
if (epoll_ctl(epfd, EPOLL_CTL_ADD, listen_sock, &ev) == -1) {
    perror("epoll_ctl: listen_sock");
    close(epfd);
    exit(EXIT_FAILURE);
}
3. 事件循环处理新连接
#define MAX_EVENTS 10
struct epoll_event events[MAX_EVENTS];

while (1) {
    int n = epoll_wait(epfd, events, MAX_EVENTS, -1);  // 阻塞等待事件
    for (int i = 0; i < n; i++) {
        if (events[i].data.fd == listen_sock) {
            // 接受新连接
            int conn_sock = accept(listen_sock, NULL, NULL);
            if (conn_sock == -1) {
                perror("accept");
                continue;
            }

            // 将新连接的 socket 添加到 epoll
            struct epoll_event ev_conn;
            ev_conn.events = EPOLLIN | EPOLLET;  // 监控可读事件
            ev_conn.data.fd = conn_sock;
            if (epoll_ctl(epfd, EPOLL_CTL_ADD, conn_sock, &ev_conn) == -1) {
                perror("epoll_ctl: conn_sock");
                close(conn_sock);
            }
        } else {
            // 处理已连接的 socket 数据
            int conn_fd = events[i].data.fd;
            char buffer[1024];
            ssize_t n = read(conn_fd, buffer, sizeof(buffer));
            if (n > 0) {
                // 处理数据...
            } else if (n == 0 || errno == ECONNRESET) {
                // 对端关闭连接,从 epoll 中删除
                epoll_ctl(epfd, EPOLL_CTL_DEL, conn_fd, NULL);
                close(conn_fd);
            }
        }
    }
}

关键注意事项

  1. 边缘触发(ET) vs 水平触发(LT)

    • ET 模式:仅在 FD 状态变化时触发事件(需一次处理完所有数据,避免饥饿)。
    • LT 模式(默认):只要满足条件,持续触发事件(编程更简单,但效率可能略低)。
  2. 错误处理

    • 检查 epoll_ctl 返回值,避免遗漏错误(如重复添加 FD 或操作已关闭的 FD)。
  3. 资源管理

    • 关闭 FD 前需从 epoll 中删除(EPOLL_CTL_DEL),否则可能导致未定义行为。
  4. 高性能优化

    • 结合非阻塞 IO 和 ET 模式实现高并发(例如 Nginx、Redis 的做法)。

总结

epoll_ctlepoll 机制的核心函数,用于动态管理监控的 FD。通过合理使用 EPOLL_CTL_ADDEPOLL_CTL_MODEPOLL_CTL_DEL 操作,可以实现高效的事件驱动网络编程,支撑数万甚至百万级的并发连接。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/968924.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

有限状态系统的抽象定义及CEGAR分析解析理论篇

文章目录 一、有限状态系统的抽象定义及相关阐述1、有限状态系统定义2、 有限状态系统间的抽象关系&#xff08;Abstract&#xff09;2.1 基于函数的抽象定义2.2 基于等价关系的抽象定义 二、 基于上面的定义出发&#xff0c;提出的思考1. 为什么我们想要/需要进行抽象2. 抽象是…

【linux学习指南】线程同步与互斥

文章目录 &#x1f4dd;线程互斥&#x1f320; 库函数strncpy&#x1f309;进程线程间的互斥相关背景概念&#x1f309;互斥量mutex &#x1f320;线程同步&#x1f309;条件变量&#x1f309;同步概念与竞态条件&#x1f309; 条件变量函数 &#x1f6a9;总结 &#x1f4dd;线…

云上话ai

这两天参加了几场ai视频直播 今天想分享一下照片&#xff0c;记录一下&#xff5e;

OpenVINO 2025.0重磅升级:开启⽣成式AI全场景⾰命!

2025年2⽉6⽇&#xff0c;英特尔OpenVINO™ 2025.0版本震撼发布&#xff0c;本次升级堪称近三年最⼤规模技术⾰新&#xff01;从⽣成 式AI性能跃升到全栈硬件⽀持&#xff0c;从开发者⼯具链优化到边缘计算突破&#xff0c;六⼤核⼼升级重新定义AI部署效率。 一&#xff0c;&a…

语言大模型基础概念 一(先了解听说过的名词都是什么)

SFT&#xff08;监督微调&#xff09;和RLHF&#xff08;基于人类反馈的强化学习&#xff09;的区别 STF&#xff08;Supervised Fine-Tuning&#xff09;和RLHF&#xff08;Reinforcement Learning from Human Feedback&#xff09;是两种不同的模型训练方法&#xff0c;分别…

裙子贴图提示词【图生图】

正向&#xff1a; (a plaid short skirt with checkered texture:1.4),(no human figure),wallpaper,incredibly absurdres,huge filesize,highres,absurdres,artbook_game c,s,rt,octane,no light,best quality,illustration,looking at viewer,impasto,canvas,realistic,rea…

【竞技宝】LCK:KT0-3爆冷不敌NS淘汰出局

北京时间2月13日&#xff0c;英雄联盟LCK2025在昨天正式迎来第一阶段的季后赛&#xff0c;首战迎来KT对阵NS&#xff0c;以下是本场比赛的详细战报。 第一局&#xff1a; KT&#xff1a;安蓓萨、大树、沙皇、韦鲁斯、布隆 NS&#xff1a;杰斯、瑟庄妮、阿萝拉、卡莎、泰坦 首…

电脑端调用摄像头拍照:从基础到实现

文章目录 1. 了解navigator.mediaDevices.getUserMedia API2. 创建 HTML 结构3. 编写 JavaScript 代码3.1 打开摄像头3.2 拍照 4. 完整代码5. 测试6. 注意事项及部署 在现代 Web 开发中&#xff0c;调用摄像头进行拍照是一个常见的功能&#xff0c;尤其是在需要用户上传头像、进…

lvs的DR模式

基于Linux的负载均衡集群软件 LVS 全称为Linux Virtual Server,是一款开源的四层(传输层)负载均衡软件 Nginx 支持四层和七层(应用层)负载均衡 HAProxy 和Nginx一样,也可同时支持四层和七层(应用层)负载均衡 基于Linux的高可用集群软件 Keepalived Keepalived是Linux…

STM32 RTC 实时时钟说明

目录 背景 RTC(实时时钟)和后备寄存器 32.768HZ 如何产生1S定时 RTC配置程序 第一次上电RTC配置 第1步、启用备用寄存器外设时钟和PWR外设时钟 第2步、使能RTC和备份寄存器访问 第3步、备份寄存器初始化 第4步、开启LSE 第5步、等待LSE启动后稳定状态 第6步、配置LSE为…

2024年12月电子学会青少年机器人技术等级考试(三级)理论综合真题

202412 青少年等级考试机器人理论真题三级 一、单选题 第 1 题 Arduino UNO/Nano主控板&#xff0c;程序模块如下&#xff0c;该模块运行后&#xff0c;引脚5输出的等效电压为0V&#xff0c;变量i对应的值是&#xff1f;&#xff08; &#xff09; A&#xff1a;0 B&#xff1…

分治中的快速排序(前序遍历)与归并排序(后序遍历)详细对比分析

目录 1. 快速排序&#xff08;前序遍历&#xff09; 核心思想与步骤 关键特性 示例分析 2. 归并排序&#xff08;后序遍历&#xff09; 核心思想与步骤 关键特性 示例分析 3. 对比总结 4. 选择依据与优化策略 5. 实际应用场景 6. 核心差异图示 7. 总结 1. 快速排序…

DeepSeek 助力 Vue 开发:打造丝滑的进度条

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

编译和链接【四】链接详解

文章目录 编译和链接【四】链接详解前言系列文章入口符号表和重定位表链接过程分段组装符号决议重定位 编译和链接【四】链接详解 前言 在我大一的时候&#xff0c; 我使用VC6.0对C语言程序进行编译链接和运行 &#xff0c; 然后我接触了VS&#xff0c; Qt creator等众多IDE&…

LeetCode每日精进:876.链表的中间结点

题目链接&#xff1a;876.链表的中间结点 题目描述&#xff1a; 给你单链表的头结点 head &#xff0c;请你找出并返回链表的中间结点。 如果有两个中间结点&#xff0c;则返回第二个中间结点。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5] 输出&#xff1a;[3,4,5…

数据结构——结构体位域、typedef类型重定义、宏、共用体union、枚举、虚拟内存划分

一、结构体位域 1.1 结构体位域的基础 结构体位域&#xff1a;把结构体字节大小扣到极致的一个类型&#xff0c;以bit单位 格式&#xff1a;struct 位域体名{数据类型 位域名:位域大小;数据类型 位域名:位域大小;...};解析&#xff1a;位域体名&#xff1a;可有可无&#xff…

CZML 格式详解,javascript加载导出CZML文件示例

示例地址&#xff1a;https://dajianshi.blog.csdn.net/article/details/145573994 CZML 格式详解 1. 什么是 CZML&#xff1f; CZML&#xff08;Cesium Zipped Markup Language&#xff09;是一种基于 JSON 的文件格式&#xff0c;用于描述地理空间数据和时间动态场景。它专…

SQL递归技巧

1.读样例 with recursive cet_dpt(id, parent_id, path, org_category, level,depart_name) as (select id ,parent_id,depart_name as path,org_category,1 as level,sd.depart_namefrom isolarerp.sys_depart sdwhere del_flag 0and sd.org_code A09B15union al…

django配置跨域

1、第一种 from django.views.decorators.csrf import csrf_exemptcsrf_exempt第二种 安装 pip install django-cors-headers在配置文件settings.py进入 INSTALLED_APPS [..."corsheaders", # 添加 ]MIDDLEWARE [corsheaders.middleware.CorsMiddleware, # 添加…

设置mysql的主从复制模式

mysql设置主从复制模式似乎很容易&#xff0c;关键在于1&#xff09;主库启用二进制日志&#xff0c;2&#xff09;从库将主库设为主库。另外&#xff0c;主从复制&#xff0c;复制些什么&#xff1f;从我现在获得的还很少的经验来看&#xff0c;复制的内容有表&#xff0c;用户…