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

网络编程——多路复用——epoll机制

理解 epoll:高效的 Linux I/O 多路复用机制

在网络编程中,处理多个并发连接是一个常见的挑战。传统的方式通常使用阻塞式 I/O 或者多线程/多进程来处理并发连接,但这些方法都存在一些性能和资源管理的问题。为了解决这些问题,Linux 引入了 epoll 这一高效的 I/O 多路复用机制。本文将详细介绍 epoll 的原理、用法和优势。

什么是 epoll?

epoll 是 Linux 操作系统提供的一种 I/O 多路复用机制,用于监视多个文件描述符的状态并进行事件驱动的 I/O 操作。它是基于事件驱动的模型,通过将文件描述符注册到 epoll 内核事件表中,然后等待内核通知有事件发生,从而避免了阻塞式 I/O 和传统的轮询方式。epoll 可以同时处理大量的并发连接,并且能够高效地处理文件描述符上的事件。

epoll 的优势

相比传统的 select 和 poll,epoll 具有以下优势:

  1. 高效:epoll 使用了更加高效的数据结构和算法,能够在大规模并发连接下提供更好的性能。
  2. 扩展性:epoll 支持水平触发和边缘触发两种模式,可以根据应用的需求选择不同的触发模式。
  3. 节省资源:epoll 使用一个文件描述符来管理多个连接,而不是每个连接都需要一个文件描述符,从而节省了资源。
  4. 没有连接数限制:epoll 没有连接数的限制,可以处理成千上万个并发连接。
  5. 高效的空间复杂度:epoll 内核事件表采用红黑树数据结构,对于大量的文件描述符,查找和插入的时间复杂度为 O(log n)。

epoll 的工作原理

epoll 的工作原理可以分为以下几个步骤:

1. 创建 epoll 实例:

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

通过调用 epoll_create() 函数创建一个 epoll 实例,返回一个文件描述符,即 epoll 文件描述符。

功能: epoll_create 函数和epoll_create1函数用于创建一个 epoll 实例(epoll 文件描述符),以用于 I/O 多路复用。
返回值: 成功时,返回一个非负整数,表示新创建的 epoll 实例的文件描述符。失败时,返回 -1,并设置全局变量 errno 表示具体的错误。
参数说明:
size:这个参数是一个整数提示,用于指定内核应为 epoll 实例分配的事件监视表的大小。然而,自 Linux 2.6.8 版本以来,这个参数被忽略,建议传递任何正数值(例如 1)。
flags:这个参数是一个整数,用于指定创建 epoll 实例的标志。

可以是以下标志之一:
EPOLL_CLOEXEC:为新文件描述符设置 close-on-exec(O_CLOEXEC)标志,表示 epoll 实例在执行 exec 系列函数时将自动关闭。这对于在子进程中避免资源泄漏很有用,例如在 fork 后。

注意: epoll_create1 函数是 epoll 系统调用的较新版本,允许设置 EPOLL_CLOEXEC 标志,以实现更好的资源管理。

#include <sys/epoll.h>int epoll_fd = epoll_create1(0);
if (epoll_fd == -1) {perror("epoll_create");return 1;
}

2.注册文件描述符:

int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event)

功能: epoll_ctl() 函数用于向 epoll 实例注册或修改文件描述符的事件。
返回值: 成功时,返回 0 表示操作成功;失败时,返回 -1,并设置全局变量 errno 表示具体的错误。
参数说明:
epfd:epoll 实例的文件描述符,即通过 epoll_create() 或 epoll_create1() 创建的 epoll 文件描述符。
op:表示要执行的操作类型,可以是以下三种操作之一:

EPOLL_CTL_ADD:向 epoll 实例添加一个文件描述符,并监视指定的事件。
EPOLL_CTL_MOD:修改 epoll 实例中已注册的文件描述符的事件。
EPOLL_CTL_DEL:从 epoll 实例中删除一个文件描述符,不再监视其事件。

fd:要操作的文件描述符,即需要注册、修改或删除的文件描述符。
event:用于指定感兴趣的事件类型和相关数据。

struct epoll_event 结构体包含两个成员:events、data
struct epoll_event {
uint32_t events; /* Epoll events /
epoll_data_t data; /
User data variable */
};
events:表示监视的事件类型,可以是以下事件之一或它们的位或运算结果:

EPOLLIN:表示文件描述符可读。
EPOLLOUT:表示文件描述符可写。
EPOLLRDHUP:表示对端关闭连接或者关闭了写入一半的连接。
EPOLLPRI:表示有紧急数据可读。
EPOLLERR:表示发生错误,如连接错误、重置等。
EPOLLHUP:表示发生挂起事件,如连接挂起、对端关闭连接等。
EPOLLET:启用边缘触发模式,即只通知状态改变的事件。
EPOLLONESHOT:一次性触发模式,即事件触发后只能被触发一次。

data:用于存放用户定义的数据,可以是任意类型的指针,通常用于记录与文件描述符相关的数据。
使用 epoll_ctl() 函数将需要监视的文件描述符注册到 epoll 实例中。可以指定感兴趣的事件类型,例如可读事件、可写事件等。

struct epoll_event ev;
ev.events = EPOLLIN; // 监视可读事件
ev.data.fd = sockfd;if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, sockfd, &ev) == -1) {perror("epoll_ctl");return 1;
}

3.等待事件:

int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout)

功能: epoll_wait() 函数用于等待 epoll 实例中的文件描述符上的事件发生,并将触发的事件填充到用户提供的数组中。
返回值: 成功时,返回触发事件的文件描述符数量,即填充到 events 数组中的事件数量;失败时,返回 -1,并设置全局变量 errno 表示具体的错误
参数说明:
epfd:epoll 实例的文件描述符,即通过 epoll_create() 或 epoll_create1() 创建的 epoll 文件描述符。
events:用于存放触发的事件的数组结构体地址,函数将把触发的事件填充到该数组中。
maxevents:表示 events 数组的最大容量,即可以存放的最大事件数量。
timeout:指定等待的超时时间(以毫秒为单位)。可以是以下值之一:

-1:表示阻塞等待,直到有事件发生为止。
0:表示立即返回,即非阻塞等待,如果没有事件发生则立即返回,不会等待。
大于0:表示等待指定的毫秒数,如果在指定时间内有事件发生则返回,否则超时返回。

使用 epoll_wait()函数等待事件的发生。当有文件描述符上的事件发生时,epoll_wait() 会返回触发事件的文件描述符信息。

#define MAX_EVENTS 10struct epoll_event events[MAX_EVENTS];while (1) {int num_events = epoll_wait(epoll_fd, events, MAX_EVENTS, -1);if (num_events == -1) {perror("epoll_wait");return 1;}// 处理事件for (int i = 0; i < num_events; ++i) {if (events[i].data.fd == sockfd) {// 可读事件发生,进行读取操作// ...}}
}

在以上代码中,我们展示了创建 epoll 实例、将文件描述符注册到 epoll 实例以及等待事件发生的过程。在实际应用中,我们可以根据具体的需求进行相应的操作和处理。这些代码片段可以作为示例帮助理解 epoll 的用法和原理。

总结:
本文介绍了 epoll 这一高效的 Linux I/O 多路复用机制,它可以有效地处理多个并发连接,并避免了传统阻塞式 I/O 的性能和资源问题。通过理解 epoll 的工作原理和用法,我们可以在网络编程中更好地应对高并发的场景,提高程序的性能和扩展性。

相关文章:

网络编程——多路复用——epoll机制

理解 epoll&#xff1a;高效的 Linux I/O 多路复用机制 在网络编程中&#xff0c;处理多个并发连接是一个常见的挑战。传统的方式通常使用阻塞式 I/O 或者多线程/多进程来处理并发连接&#xff0c;但这些方法都存在一些性能和资源管理的问题。为了解决这些问题&#xff0c;Lin…...

chapter14:springboot与安全

Spring Boot与安全视频 Spring Security, shiro等安全框架。主要功能是”认证“和”授权“&#xff0c;或者说是访问控制。 认证&#xff08;Authentication&#xff09;是建立在一个声明主体的过程&#xff08;一个主体一般指用户&#xff0c;设备或一些可以在你的应用程序中…...

Linux初识网络基础

目录 网络发展 认识“协议 ” 网络协议 OSI七层模型&#xff1a; TCP/IP五层&#xff08;或四层&#xff09;模型 网络传输基本流程 网络传输流程图&#xff1a; 数据包封装和封用 网络中的地址 认识IP地址&#xff1a; 认识MAC地址&#xff1a; 网络发展 1.独立…...

vue3+ts 动态导入多文件组件

1、在components文件夹中新建index.ts文件&#xff08;components文件夹下为创建的组件&#xff09; // index.ts const modules import.meta.globEager("./*.vue"); //参数为组件路径 let componentsOpts {}const getCaption (obj, str, z: boolean) > {let…...

补充122836356

ul {margin-bottom: 20px;& > li {margin-bottom: 0;} }等效于ul {margin-bottom: 20px; }ul > li {margin-bottom: 0; }CSS中的&代表的什么 如源码&#xff1a; ul{ margin-bottom: 20px; & >li { margin-bottom: 0; } } & 表示嵌套的上一级 这是s…...

记录 pl-table 表格头部文字抖动的问题

本文记录一个实际开发中 pl-table 的问题&#xff0c;项目比较老&#xff0c;vue还是2.x版本。pl-table 是基于 el-table 改造过来的表格展示组件&#xff0c;已经停止更新。 问题描述 当 data 内数据动态改变时&#xff0c;pl-table 的表头部分&#xff0c;列的文字会左右抖动…...

Vite 创建 Vue项目之后,eslint 错误提示的处理

使用 npm create vuelatest创建 vue 项目&#xff08;TS&#xff09;之后&#xff0c;出现了一些 eslint 错误提示&#xff0c;显然&#xff0c;不是代码真实的错误&#xff0c;而是提示搞错了。 vuejs/create-vue: &#x1f6e0;️ The recommended way to start a Vite-pow…...

FFmpeg 硬编码VideoToolBox流程

介绍 FFmpeg已经提供对 VideoToolBox 的编解码支持&#xff1b;主要涉及到的文件有videotoolbox.c、videotoolbox.h、videotoolboxenc.c、ffmepg_videotoolbox.c。在编译 FFmpeg 源码时&#xff0c;想要支持VideoToolBox&#xff0c;在 configure 时&#xff0c;需要–enable-…...

恒盛策略:内盘是买入还是卖出?

内盘&#xff0c;又称成交明细&#xff0c;是指交易所实时发布的每一笔成交价格、交易时刻、成交量等具体数据。关于股民来说&#xff0c;每天重视内盘数据已成为必修课。可是&#xff0c;当看到内盘数据时&#xff0c;很多人都会困惑&#xff0c;到底内盘是买入还是卖出呢&…...

安装Lombok--Lombok的常用注解说明及使用方法

&#x1f600;前言 本篇博文是关于Lombok的基本介绍和基本使用&#xff0c;希望能够帮助到您&#x1f60a; &#x1f3e0;个人主页&#xff1a;晨犀主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是晨犀&#xff0c;希望我的文章可以帮助到大家&#xff0c;您的满…...

无涯教程-Perl - endpwent函数

描述 此功能告诉系统您不再希望使用getpwent从密码文件读取条目。在Windows下,使用Win32API::Net函数从域服务器获取信息。 语法 以下是此函数的简单语法- endpwent返回值 此函数不返回任何值。 例 以下是显示其基本用法的示例代码- #!/usr/bin/perlwhile(($name, $pas…...

vue项目在body设置公共的背景前提下,区分首页背景图和其他页面背景图

1.需求:在vue项目已设置统一的body背景图的前提,单独给首页换一个背景图,然后其他页面背景图不变的临时需求 实现思路1:在首页home.vue中 在公共的style.css文件中写上两个背景样式(写在公共样式中是因为style.css比组件内部的先加载,避免页面出现后背景空白的问题) …...

测试人员该怎样写软件缺陷报告?

软件测试过程中&#xff0c;每个公司都制订了软件的缺陷处理流程&#xff0c;每个公司的软件缺陷处理流程不尽相同&#xff0c;但是它们遵循的最基本流程是一样的&#xff0c;都要经过提交、分配、确认、处理、复测、关闭等环节&#xff0c;如图1所示。 缺陷处理流程 关于图1所…...

【大数据】Flink 详解(二):核心篇 Ⅱ

Flink 详解&#xff08;二&#xff09;&#xff1a;核心篇 Ⅱ 22、刚才提到 State&#xff0c;那你简单说一下什么是 State。 在 Flink 中&#xff0c;状态 被称作 state&#xff0c;是用来保存中间的计算结果或者缓存数据。根据状态是否需要保存中间结果&#xff0c;分为 无状…...

一孩半政策

一&#xff09; 一孩半&#xff0c;又称独女户二胎&#xff0c;即中国大陆部分农村的一项计划生育政策&#xff0c;第一胎是女孩的夫妻可生育第二个子女。试问这个政策会不会影响男女平衡。 二&#xff09;如果生女孩一直生&#xff0c;直到生男孩停止&#xff0c;试问会不会…...

如何在 Spring Boot 中集成日志框架 SLF4J、Log4j

文章目录 具体步骤附录 笔者的操作环境&#xff1a; Spring Cloud Alibaba&#xff1a;2022.0.0.0-RC2 Spring Cloud&#xff1a;2022.0.0 Spring Boot&#xff1a;3.0.2 Nacos 2.2.3 Maven 3.8.3 JDK 17.0.7 IntelliJ IDEA 2022.3.1 (Ultimate Edition) 具体步骤 因为 …...

如何在Linux布置nginx(附带Nginx基本操作步骤)

文章目录 前言一、下载环境依赖二、下载nginx安装包三、具体操作流程总结 前言 提示&#xff1a;下述操作步骤适合内网服务器、局域网服务器和公网服务器。 不足之处欢迎交流指正&#xff0c;不喜勿喷。 一、下载环境依赖 yum -y install gcc zlib zlib-devel pcre-devel ope…...

Xcode升级导致关联库报错

想办法找到对应的库 然后到 Build Phases -- LinkBinary With Libraries中点击&#xff0c;选择对应的framework即可&#xff0c;就像我工程的报错 Undefined symbol: _OBJC_CLASS_$_ADClient _OBJC_CLASS_$_ASIdentifierManager 缺失的库是AdSupport.framework 添加后再次编…...

利用docker run --rm 命令实现使用宿主机中没有的命令

利用docker run --rm 命令实现使用宿主机中没有的命令 使用容器中的jar命令解压jar包&#xff0c;并将解压内容输出到挂载在宿主机中的目录里使用宿主机中没有的nmap命令来通过端口找IP 使用容器中的jar命令解压jar包&#xff0c;并将解压内容输出到挂载在宿主机中的目录里 do…...

中级课程——XSS

文章目录 介绍挖掘思路分类反射型存储型dom类型 介绍 挖掘思路 注入点&#xff1a;各种输入框 测试代码&#xff08;poc&#xff09;&#xff1a;js语句 分类 反射型 存储型 dom类型...

模型参数、模型存储精度、参数与显存

模型参数量衡量单位 M&#xff1a;百万&#xff08;Million&#xff09; B&#xff1a;十亿&#xff08;Billion&#xff09; 1 B 1000 M 1B 1000M 1B1000M 参数存储精度 模型参数是固定的&#xff0c;但是一个参数所表示多少字节不一定&#xff0c;需要看这个参数以什么…...

Objective-C常用命名规范总结

【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名&#xff08;Class Name)2.协议名&#xff08;Protocol Name)3.方法名&#xff08;Method Name)4.属性名&#xff08;Property Name&#xff09;5.局部变量/实例变量&#xff08;Local / Instance Variables&…...

CentOS下的分布式内存计算Spark环境部署

一、Spark 核心架构与应用场景 1.1 分布式计算引擎的核心优势 Spark 是基于内存的分布式计算框架&#xff0c;相比 MapReduce 具有以下核心优势&#xff1a; 内存计算&#xff1a;数据可常驻内存&#xff0c;迭代计算性能提升 10-100 倍&#xff08;文档段落&#xff1a;3-79…...

Python爬虫(二):爬虫完整流程

爬虫完整流程详解&#xff08;7大核心步骤实战技巧&#xff09; 一、爬虫完整工作流程 以下是爬虫开发的完整流程&#xff0c;我将结合具体技术点和实战经验展开说明&#xff1a; 1. 目标分析与前期准备 网站技术分析&#xff1a; 使用浏览器开发者工具&#xff08;F12&…...

【Zephyr 系列 10】实战项目:打造一个蓝牙传感器终端 + 网关系统(完整架构与全栈实现)

🧠关键词:Zephyr、BLE、终端、网关、广播、连接、传感器、数据采集、低功耗、系统集成 📌目标读者:希望基于 Zephyr 构建 BLE 系统架构、实现终端与网关协作、具备产品交付能力的开发者 📊篇幅字数:约 5200 字 ✨ 项目总览 在物联网实际项目中,**“终端 + 网关”**是…...

tree 树组件大数据卡顿问题优化

问题背景 项目中有用到树组件用来做文件目录&#xff0c;但是由于这个树组件的节点越来越多&#xff0c;导致页面在滚动这个树组件的时候浏览器就很容易卡死。这种问题基本上都是因为dom节点太多&#xff0c;导致的浏览器卡顿&#xff0c;这里很明显就需要用到虚拟列表的技术&…...

管理学院权限管理系统开发总结

文章目录 &#x1f393; 管理学院权限管理系统开发总结 - 现代化Web应用实践之路&#x1f4dd; 项目概述&#x1f3d7;️ 技术架构设计后端技术栈前端技术栈 &#x1f4a1; 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 &#x1f5c4;️ 数据库设…...

保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek

文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama&#xff08;有网络的电脑&#xff09;2.2.3 安装Ollama&#xff08;无网络的电脑&#xff09;2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...

Redis:现代应用开发的高效内存数据存储利器

一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发&#xff0c;其初衷是为了满足他自己的一个项目需求&#xff0c;即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源&#xff0c;Redis凭借其简单易用、…...

「全栈技术解析」推客小程序系统开发:从架构设计到裂变增长的完整解决方案

在移动互联网营销竞争白热化的当下&#xff0c;推客小程序系统凭借其裂变传播、精准营销等特性&#xff0c;成为企业抢占市场的利器。本文将深度解析推客小程序系统开发的核心技术与实现路径&#xff0c;助力开发者打造具有市场竞争力的营销工具。​ 一、系统核心功能架构&…...