当前位置: 首页 > news >正文

深入剖析Linux网络IO与epoll

epoll的优点

(1)不需要轮询所有的文件描述符 (2)每次取就绪集合,都在固定位置 (3)事件的就绪和IO触发可以异步解耦

四、epoll函数原型

4.1、epoll_create(int size)

代码语言:javascript

AI代码解释

#include <sys/epoll.h> int epoll_create(int size);

功能:创建epoll的文件描述符。 参数说明:size表示内核需要监控的最大数量,但是这个参数内核已经不会用到,只要传入一个大于0的值即可。 当size<=0时,会直接返回不可用,这是历史原因保留下来的,最早的epoll_create是需要定义一次性就绪的最大数量;后来使用了链表以便便维护和扩展,就不再需要使用传入的参数。 返回:返回该对象的描述符,注意要使用 close 关闭该描述符。

4.2、epoll_ctl

代码语言:javascript

AI代码解释

#include <sys/epoll.h> int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event); // epoll_ctl对应系统调用sys_epoll_ctl

功能:操作epoll的文件描述符,主要是对epoll的红黑树节点进行操作,比如节点的增删改查。 参数说明:

参数

含义

epfd

通过 epoll_create 创建的文件描述符

op

对红黑树的操作,比如节点的增加、修改、删除,分别对应EPOLL_CTL_ADD、EPOLL_CTL_MOD、EPOLL_CTL_DEL

fd

需要添加监听的文件描述符

event

事件信息

4.2.1、event参数说明

struct epoll_event结构体原型

代码语言:javascript

AI代码解释

typedef union epoll_data{ void* ptr; int fd; uint32_t u32; uint64_t u64 }; struct epoll_event{ uint32_t events; epoll_data_t data; }

events成员代表要监听的epoll事件类型 events成员:

成员变量

含义

EPOLLIN

监听fd的读事件

EPOLLOUT

监听fd的写事件

EPOLLRI

监听紧急数据可读事件(带外数据到来)

EPOLLRDHUP

监听套接字关闭或半关闭事件

EPOLLET

将EPOLL设为边缘触发(Edge Triggered)模式

data成员: data 成员是一个联合体类型,可以在调用 epoll_ctl 给 fd 添加/修改描述符监听的事件时携带一些数据,方便后面的epoll_wait可以取出信息使用。

4.2.2、扩展说明:SYSCALL_DEFINE数字 的宏定义

跟着的数字代表函数需要的参数数量,比如SYSCALL_DEFINE1代表函数需要一个参数、SYSCALL_DEFINE4代表函数需要4个参数。

4.2.3、注意

epoll_ctl是非阻塞的,不会被挂起。

4.3、epoll_wait

函数原型

代码语言:javascript

AI代码解释

#include <sys/epoll.h> int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout);

功能:阻塞一段时间,等待事件发生 返回:返回事件数量,事件集添加到events数组中。也就是遍历红黑树中的双向链表,把双向链表中的节点数据拷贝出来,拷贝完毕后把节点从双向链表中移除。

返回值

含义

大于0

事件个数

等于0

超时时间timeout到了

小于0

出错,可通过errno查看出错原因

参数说明:

参数

含义

epfd

通过 epoll_create 创建的文件描述符

events

存放就绪的事件集合,是输出参数

maxevents

最大可存放事件数量,events数组大小

timeout

阻塞等待的时间长短,单位是毫秒,-1表示一直阻塞等待

五、epoll使用步骤

图片

step 1:创建epoll文件描述符

代码语言:javascript

AI代码解释

int epfd = epoll_create(1);

step 2:创建struct epoll_event结构体

代码语言:javascript

AI代码解释

struct epoll_event ev; ev.data.fd=listenfd;//保存监听的fd,以便epoll_wait的后续操作 ev.events=EPOLLIN;//设置监听fd的可读事件

step 3:添加事件监听

代码语言:javascript

AI代码解释

epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,&ev);

step 4:等待事件

代码语言:javascript

AI代码解释

struct epoll_event events[EVENTS_LENGTH]; char rbuffer[MAX_BUFF]={ 0 }; char wbuffer[MAX_BUFF]={ 0 }; while(1) { int nready = epoll_wait(epfd,events,EVENTS_LENGTH,-1);//-1表示阻塞等待 int i=0; for(i=0;i<nready;i++) { int clientfd=events[i].data.fd; if(clientfd==listenfd) { struct sockaddr_in client; int len=sizeof(client); int confd=accept(listenfd,(struct sockaddr*)&client,&len); //step 2:创建struct epoll_event结构体 struct epoll_event evt; evt.data.fd=confd;//保存监听的fd,以便epoll_wait的后续操作 evt.events=EPOLLIN;//设置监听fd的可读事件 // step 3:添加事件监听 epoll_ctl(epfd,EPOLL_CTL_ADD,confd,&evt); } else if(events[i].events &EPOLLIN) { int ret = recv(clientfd,rbuffer,MAX_BUFF,0); if(ret>0) { rbuffer[ret]='\0';//剔除干扰数据 printf("recv: %s\n",rbuffer); memcpy(wbuffer,rbuffer,MAX_BUFF);//拷贝数据,做回传示例 //step 2:创建struct epoll_event结构体 struct epoll_event evt; evt.data.fd=clientfd;//保存监听的fd,以便epoll_wait的后续操作 evt.events=EPOLLOUT;//设置监听fd的可写事件 // step 3:修改事件监听 epoll_ctl(epfd,EPOLL_CTL_MOD,clientfd,&evt); } } else if(events[i].events &EPOLLOUT) { int ret = send(clientfd,wbuffer,MAX_BUFF,0); printf("send: %s\n",wbuffer); //step 2:创建struct epoll_event结构体 struct epoll_event evt; evt.data.fd=clientfd;//保存监听的fd,以便epoll_wait的后续操作 evt.events=EPOLLIN;//设置监听fd的可读事件 // step 3:修改事件监听 epoll_ctl(epfd,EPOLL_CTL_MOD,clientfd,&evt); } } }

六、完整示例代码

代码语言:javascript

AI代码解释

#include <stdio.h> #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <fcntl.h> #include <unistd.h> #include <pthread.h> #include <sys/epoll.h> #include <string.h> #define BUFFER_LENGTH 128 #define EVENTS_LENGTH 128 char rbuff[BUFFER_LENGTH] = { 0 }; char wbuff[BUFFER_LENGTH] = { 0 }; int main() { // block int listenfd = socket(AF_INET, SOCK_STREAM, 0); // if (listenfd == -1) return -1; // listenfd struct sockaddr_in servaddr; servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = htonl(INADDR_ANY); servaddr.sin_port = htons(9999); if (-1 == bind(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr))) { return -2; } #if 0 // nonblock int flag = fcntl(listenfd, F_GETFL, 0); flag |= O_NONBLOCK; fcntl(listenfd, F_SETFL, flag); #endif listen(listenfd, 10); int epfd = epoll_create(1); struct epoll_event ev, events[EVENTS_LENGTH]; ev.events = EPOLLIN; ev.data.fd = listenfd; epoll_ctl(epfd, EPOLL_CTL_ADD, listenfd, &ev); printf("epfd : %d\n", epfd); while (1) { int nready = epoll_wait(epfd, events, EVENTS_LENGTH, -1); printf("nready --> %d\n",nready); int i; for (i = 0; i < nready;i++) { int clientfd = events[i].data.fd; if (listenfd == clientfd) { // accept struct sockaddr_in client; int len = sizeof(client); int conffd = accept(clientfd, (struct sockaddr*)&client,&len); printf("conffd --> %d\n",conffd); ev.events = EPOLLIN; ev.data.fd = conffd; epoll_ctl(epfd, EPOLL_CTL_ADD, conffd, &ev); } else if(events[i].events & EPOLLIN)//client { int ret=recv(clientfd, rbuff, BUFFER_LENGTH, 0); if (ret > 0) { rbuff[ret] = '\0'; printf("recv buffer: %s\n", rbuff); /* int j; for (j = 0; j < BUFFER_LENGTH;j++) { buff[j] = 'a' + (j % 26); } send(clientfd, buff, BUFFER_LENGTH, 0); */ memcpy(wbuff, rbuff, BUFFER_LENGTH); ev.events = EPOLLOUT; ev.data.fd = clientfd; epoll_ctl(epfd, EPOLL_CTL_MOD, clientfd, &ev); } } else if (events[i].events & EPOLLOUT) { send(clientfd, wbuff, BUFFER_LENGTH, 0); printf("send --> %s\n",wbuff); ev.events = EPOLLIN; ev.data.fd = clientfd; epoll_ctl(epfd, EPOLL_CTL_MOD, clientfd, &ev); } } } return 0; }

七、epoll的缺点

读写使用相同的缓冲区。比如上述的示例中,wbuffer和rbuffer是使用同一个缓冲区的,所以需要rbuff[ret] = '\0';去除杂数据。

http://www.jsqmd.com/news/813298/

相关文章:

  • 小红书作品一键下载神器:XHS-Downloader 终极使用指南
  • 2026年靠谱的316L不锈钢管/大口径不锈钢管/焊接不锈钢管公司选择指南 - 品牌宣传支持者
  • 电池电动汽车(BEV)核心技术解析:从成本拐点到产业链重构
  • 开源多媒体中心MythTV:模块化架构与家庭媒体服务器实践
  • 别再被Excel文件‘炸’了!手把手教你用ZipSecureFile.setMinInflateRatio解决Apache POI的Zip Bomb报错
  • 直流无刷电机厂家哪家好?认准恒驱!专业定制微型、关节、减速电机,汽车座椅、割草机电机专业供应商 - 栗子测评
  • 开源机械臂与Home Assistant集成:打造可交互的智能家居物理终端
  • C语言向C++过渡
  • 2026年4月餐饮底料品牌推荐,美蛙鱼底料/冷锅鱼底料/火锅底料/餐饮底料/底料/鱼蛙火锅底料,餐饮底料批发厂家咨询热线 - 品牌推荐师
  • 优质焊条烘箱供应商有哪些?正规焊剂烘箱定制厂家|2026年实力厂家盘点与推荐:莱豪热处理领衔 - 栗子测评
  • shell 脚本中 case 语句的语法错误如何排查?
  • AI算力需求爆发,光模块龙头中际旭创成创业板第二只千元股,王伟修家族身家飙升!
  • AI智能体商业化实战:x402支付技能包集成指南
  • 告别编译烦恼!UE4/UE5开发者必备:开箱即用的CEF3(支持H.264/MP4)替换包使用指南
  • 校企合作奖学金与实习计划:破解半导体硬件人才困境的务实路径
  • 从Claude Code频繁封号到稳定使用Taotoken接入的体验对比
  • 点云匹配方法 NDT(正态分布变换)
  • 珠三角保安公司甄选指南!惠州东莞深圳广州佛山保安公司对比,惠州工厂保安公司优选认准广东国卫保安 - 栗子测评
  • yargs单元测试终极指南:使用mocha测试CLI命令的完整实践
  • 杭州锦纪财务咨询有限公司2026一站式财税优选:杭州工商注册/记账报税/代办营业执照/出口退税代理推荐杭州锦纪财务 - 栗子测评
  • 基于AutoHotkey v2的Cursor AI编程效率工具:CapsLock快捷键方案详解
  • 从绿度到热度:拆解RSEI遥感生态指数的四个核心指标(GEE/Landsat 8版)
  • 【独家首发】DeepSeek-V2模型GPU利用率可视化方案:仅需3个自定义Metrics,告别盲调参数
  • PX4-Autopilot扩展卡尔曼滤波状态估计系统深度解析与实战调优
  • 深入解析Baichuan-7B:从GPT架构到LoRA微调的实践指南
  • 2026年卫生级拉丝白钢板/316L白钢板/321白钢板/沈阳拉丝白钢板推荐厂家精选 - 品牌宣传支持者
  • 2026年测试工程师常用性能测试平台:高效办公与场景适配指南
  • 【坐标转换实战】从公式到代码:极坐标与笛卡尔坐标互通的编程实现与象限陷阱
  • 联发科与威睿电通合作:深度解析全球模式SoC如何实现CDMA与LTE融合
  • 三步轻松上手:BilldDesk Pro开源远程桌面控制工具完整指南