前端 详解epoll_events结构体

2024-04-12 01:32

本文主要是介绍前端 详解epoll_events结构体,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Epoll 作为一种 IO 复用机制多应用与高并发领域,网上有很多如何使用 epoll 的基础教程,但对于 epoll 中很重要的结构体 epoll_event 讲的都模棱两可,这篇文章将做深入解析

在解析之前,先回顾一下 epoll 的使用方法。

  • 首先调用int epoll_create(int size);创建一个 epoll
  • 调用int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);为 epoll 注册事件(如果是新建的 epoll 一般 op 选项是EPOLL_CTL_ADD添加事件)
  • 调用int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout);等待事件的到来,得到的结果存储在 event 中
  • 完全处理完毕后,再次调用int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);删除已经注册的事件(op 选项是EPOLL_CTL_DEL

值得注意的是epoll_wait函数只能获取是否有注册事件发生,至于这个事件到底是什么、从哪个 socket 来、发送的时间、包的大小等等信息,统统不知道。这就好比一个人在黑黢黢的山洞里,只能听到声响,至于这个声音是谁发出的根本不知道。因此我们就需要struct epoll_event来帮助我们读取信息。

2.struct 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_event 结构体的定义如上所示,分为 events 和 data 两个部分。

events 是 epoll 注册的事件,比如EPOLLINEPOLLOUT等等,这个参数在epoll_ctl注册事件时,可以明确告知注册事件的类型。

第二个参数 data 是一个联合体,很多人搞不清除 data 拿来干嘛,网上给的解释一般是传递参数,至于怎么传?有什么用?都不清不楚。下面一个小节将用实例的方式分析。

3.struct epoll_event 使用实例

下面将从两个实际案例中,分析 epoll_event 的作用。

3.1 示例 1:服务器侦听客户端连接

//创建socket
nSocketListen = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
...
//绑定地址
struct sockaddr_in local;
memset(&local, 0, sizeof(local));
local.sin_family = AF_INET;
local.sin_addr.s_addr = htonl(INADDR_ANY);//0.0.0.0所有地址都合法
local.sin_port = htons(TCP_PORT);
bind(nSocketListen, (struct sockaddr*) & local, sizeof(local))

//创建epoll nListenEpoll = epoll_create(MAX_LISTEN_EVENTS); //注册事件 struct epoll_event Ev; memset(&Ev, 0, sizeof(epoll_event)); Ev.events= EPOLLIN | EPOLLET Ev.data.fd = nSocketListen;

epoll_ctl(nListenEpoll, EPOLL_CTL_ADD, nSocketListen, &Ev); //侦听 int nFdNumber = epoll_wait(nListenEpoll, lpListenEvents, MAX_LISTEN_EVENTS, -1); //处理侦听结果 for (int i = 0; i < nFdNumber; i++) { if (lpListenEvents[i].data.fd != nSocketListen) continue; ... }

上述代码是网上很常见的 demo 片段,作用是建立一个服务器,侦听所有客户端的连接。具体过程是先建立了一个 socket,地址设为设为 0.0.0.0(所有人都可以连接),然后将这个 socket 的句柄 nSocketListen 附加在注册事件Ev.data.fd上。在 wait 等到结果后做一个判断,看看接收到和预设的是否一致

if (lpListenEvents[i].data.fd != nSocketListen) continue;

这里联合体 data 中的 fd 起到了传递 socket 句柄的作用,这样我们就知道:等到的事件是不是我们想要的 socket 产生的。但是这个网上很常见的 demo 其实并没有体现出 fd 传参的作用!

整个程序仅仅设置并注册了一个 socket 来连接所有 IP 地址htonl(INADDR_ANY);,因此 wait 收到的消息必然来自于这个唯一的 socket,所以这句判断根本是多此一举。

正确的使用方法是:我们可以建立三个 socket 管理不同的字段

Socket 句柄管理的 IP 地址范围
101100-120
102121-191
103192-255

将这三个 socket 都注册进 epoll 里面,当 wait 到来时,我们就可以根据Ev.data.fd传进来的 socket 句柄来进行处理。比如上午 8 点到 10 点这个时间段,服务器只允许 100-120 范围的 IP 连接进来,就可以做一个判断if (lpListenEvents[i].data.fd == 101),如果是再接受连接。

这个例子中,fd 传递了 socket 的句柄,帮助我们管理不同的网络连接。

3.2 示例 2:线程间通信

//线程A代码
struct epoll_event Ev;
memset(&Ev, 0, sizeof(Ev));
Ev.events= EPOLLOUT | EPOLLET | EPOLLERR | EPOLLHUP
Ev.data.ptr = lpCatList;

epoll_ctl(iClientEpoll, EPOLL_CTL_ADD, lpCatList->nClientSocket, &Ev);

//线程B代码
int nFdNumber = epoll_wait(iClientEpoll, lpEvent, MAX_CLIENT_EVENTS, -1);IOPACKHEAD_LIST* RelpCatList = (IOPACKHEAD_LIST*)lpEvent[i].data.ptr;

上述 demo 展示了 epoll 在两个线程间协同工作。线程 A 功能相当于接线员,跟 3.1 节展示的服务器功能相同:监听客户的连接,accept 客户的请求,建立客户与服务器间的 socket 连接通道(此处的建立的 socket 句柄为 nClientSocket)。然后将这些客户连接注册到 iClientEpoll 中

这些通道建立后,客户一般不会时刻收发数据,也就是说客户可能不定时的使用为他们建立的 socket 连接通道,线程 B 的 iClientEpoll 就是用来监听有没有已经建立连接的客户需要收发数据的。

如果仅仅像 3.1 节所展示的那样用Ev.data.fd传一个客户 socket 的句柄,这样线程 B 能得到的信息太少了。所以我们需要使用结构体 lpCatList 来传参。

lpCatList 相当于一个令牌,他是一个指针,指向的地址存储了客户的信息(Socket 句柄,IP 地址,MAC 地址,请求时间等等),A 线程在接收客户连接后,将他们写到这个令牌中,一并注册到 iClientEpoll。B 线程就可以利用 Ev.data.ptr 包含的重要的地址信息。

这样 ptr 就相当于一个小纸条,A 线程通过 iClientEpoll 将这个小纸条交到 B 线程手中,B 线程就能了解 A 线程的信息,实现了线程间的通信。

下面我们打印一下线程 A 的 lpCatlist

(gdb) p lpCatList
$18 = (IOPACKHEAD_LIST *) 0x7ffff0001120

再打印一下线程 B 的 ptr,可以发现他们指向同一个地址 0x7ffff0001120,说明参数成功传递

(gdb) p lpEvent[0]
$14 = {events = 4, data = {ptr = 0x7ffff0001120, fd = -268431072, u32 = 4026536224, u64 = 140737219924256}}

详解epoll_events结构体 · 大专栏 (dazhuanlan.com)

这篇关于前端 详解epoll_events结构体的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/895728

相关文章

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

java中反射(Reflection)机制举例详解

《java中反射(Reflection)机制举例详解》Java中的反射机制是指Java程序在运行期间可以获取到一个对象的全部信息,:本文主要介绍java中反射(Reflection)机制的相关资料... 目录一、什么是反射?二、反射的用途三、获取Class对象四、Class类型的对象使用场景1五、Class