【Linux网络编程】IO复用技术 select、poll、epoll函数使用 服务器/客户端举例

目录

前言

一、IO复用基本概念

💻阻塞I/O

💻非阻塞I/O

💻I/O复用

📌select  函数 

📌poll  函数

📌epoll  函数

二、利用I/O复用搭建服务器/客户端

💻服务器完整代码

💻客户端完整代码

💻测试效果 


前言

本文主要学习Linux内核编程,结合Visual Studio 2019进行跨平台编程,内容包括IO复用技术相关知识介绍以及用服务器客户端实例说明(详细举例epoll的使用 )

一、IO复用基本概念

💻阻塞I/O

最流行的I/O模型是阻塞I/O模型,缺省时,所有的套接口都是阻塞的

💻非阻塞I/O

当我们把一个套接口设置为非阻塞方式时,即通知内核,当请求的I/O操作非得让进程睡眠不能完成时,不要让进程睡眠,而应返回一个错误

应用程序连续不断地查询内核,看看某操作是否准备好,这对CPU时间是极大的浪费,一般只在专门提供某种功能的系统中才会用到

💻I/O复用

有了I/O复用,我们就可以调用select或poll,在这两个系统调用的某一个上阻塞,而不是真正阻塞于真正的I/O系统调用

如果一个或多个I/O条件满足(例如:输入已准备好被读,或者描述字可以承接更多输出的时候)

我们就能够被通知到,这样的能力被称为I/O复用,是由函数select和poll支持的

I/O复用网络应用场合:

  • 当客户处理多个描述字
  • 一个客户同时处理多个套接口
  • 如果一个TCP服务器既要处理监听套接口,又要处理连接套接口
  • 如果一个服务器既要处理TCP,又要处理UDP 

📌select  函数 

select函数允许进程指示内核等待多个事件中的任一个发生,并仅在一个或多个事件发生或经过某指定的时间后才唤醒进程

📍头文件:<sys/select.h>  <sys/socket.h>

📍作用:  提供了即时响应多个套接的读写事件

📍原型:

int select(int maxfdp1,fd_set *readset,fd_set *writeset,fd_set *except,

const struct timeval *timeout);

📍参数:

maxfdp1: 等待最大套接字值加1(等待套接字的数量)

readset:要检查读事件的容器

writeset:要检查写事件的容器

timeout:  超时时间

📍返回值: 返回触发套件接字的个数

📌poll  函数

poll函数和select类似,但它是用文件描述符而不是条件的类型来组织信息

也就是说,一个文件描述符的可能事件都存储在struct pollfd中

与之相反,select用事件的类型来组织信息,而且读,写和错误情况都有独立的描述符掩码

poll函数是POSIX:XSI扩展的一部分,它起源于UNIX System V 

📍头文件:  <poll.h>

📍作用:  提供了即时响应多个套接的读写事件

📍原型:

int poll(struct pollfd *fdarray,unsigned long nfds,int timeout);

📍参数:

fdarray:是一个pollfd的机构体数组用来表示表示文件描述符的监视信息

<think>嗯,用户让我详细介绍一下I/O多路复用中select的用法。首先,我需要回忆一下select的基本概念和作用。select是一种I/O多路复用机制,允许程序同时监视多个文件描述符,看是否有可读、可写或异常事件发生。这样可以在单线程中处理多个I/O操作,提高效率。 接下来,我应该从select的原理开始解释。用户可能需要了解为什么需要select,比如在同时处理多个socket连接时,避免为每个连接创建一个线程,从而减少资源消耗。不过要注意不要讲得太复杂,保持简洁明了。 然后,需要介绍select函数原型,参数含义。记得提到nfds是最大的文件描述符加1,三个fd_set参数分别对应读、写、异常事件,timeout是超时时间。这部分要详细,但也要用例子帮助理解,比如fd_set的使用步骤,FD_ZERO、FD_SET这些宏的作用。 然后,使用步骤应该分点说明,比如初始化fd_set,添加描述符,调用select,然后检查哪些描述符就绪。每个步骤要具体,可能还要给出示例代码片段,但用户可能需要的是解释而不是代码,所以用文字描述过程更好。 接着,可能需要讨论select的优缺点。优点包括跨平台支持,缺点有最大文件描述符限制、效率随数量增加下降、需要每次重新设置fd_set等。这部分要客观,指出虽然select有局限,但在某些场景下仍然适用。 然后,给出一个简单的应用场景,比如TCP服务器监听多个客户端连接。用这个例子说明select如何检测新连接和客户端数据到达。用户可能更理解实际应用的例子,这样能帮助他们将理论联系实际。 最后,注意事项部分要提醒用户关于fd_set的大小限制,比如在Linux下通常是1024,以及处理异常情况的重要性。可能还要提到select与其他I/O多路复用技术pollepoll的区别,但不需要展开,保持重点在select上。 整个过程中要确保使用中文,结构清晰,分点列出,符合用户要求的系统级指令中的格式要求,比如行内公式用$...$,但在这个问题中可能用不上数学公式,不过要注意其他格式要求,比如单独成段的独立公式,如果有的话。但在这个回答中可能不需要数学公式,所以重点放在文字解释上。 需要检查是否有LaTeX语法错误,但这里可能用到的LaTeX不多,不过如果有代码片段,可能需要用等宽字体或者代码块,但用户没有特别要求,所以用自然语言描述即可。同时要保证回答真实可靠,比如准确描述select的参数和步骤,不出现错误信息。 可能用户是刚开始学习网络编程的开发者,或者需要处理高并发连接的工程师,他们需要了解基本的select用法来优化他们的程序。因此,解释时要避免过于高深,保持基础但全面。同时,用户可能想知道在实际项目中如何应用,所以举例很重要。 总结一下,回答的结构应该是:原理简介、函数原型详解、使用步骤、优缺点分析、应用示例、注意事项。每个部分分点说明,使用标题和子标题,保持层次分明。这样用户能够逐步理解select的用法,并能够应用到实际开发中。</think>### I/O多路复用中 `select` 用法详解 #### 一、基本原理 `select` 是一种**同步I/O多路复用**机制,通过监视多个文件描述符(如套接字)的状态(可读、可写、异常),实现在单线程中同时处理多个I/O事件。核心思想是**将多个阻塞操作集中管理**,避免为每个I/O操作单独创建线程。 --- #### 二、函数原型 ```c #include <sys/select.h> int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout); ``` - **参数说明**: - `nfds`:监视的文件描述符最大值 + 1(例如监视0~4号描述符,则`nfds=5`)。 - `readfds`:需检测**可读事件**的文件描述符集合。 - `writefds`:需检测**可写事件**的文件描述符集合。 - `exceptfds`:需检测**异常事件**的文件描述符集合。 - `timeout`:超时时间。若为`NULL`则阻塞等待;若为0则立即返回。 - **返回值**: - 成功时返回就绪的文件描述符总数。 - 超时返回0,错误返回-1。 --- #### 三、使用步骤 1. **初始化文件描述符集合** 使用 `FD_ZERO` 清空集合,`FD_SET` 添加需监视的描述符: ```c fd_set read_fds; FD_ZERO(&read_fds); // 清空集合 FD_SET(sockfd, &read_fds); // 添加sockfd到读集合 ``` 2. **调用 `select`** 传入集合并等待事件就绪: ```c int ret = select(max_fd + 1, &read_fds, NULL, NULL, &timeout); ``` 3. **检查就绪事件** 使用 `FD_ISSET` 判断具体描述符是否就绪: ```c if (FD_ISSET(sockfd, &read_fds)) { // 处理sockfd的可读事件 } ``` --- #### 四、优缺点分析 - **优点**: - 跨平台支持(Linux/Windows/macOS)。 - 简单易用,适合少量并发场景。 - **缺点**: - 文件描述符数量受限(默认1024)。 - 每次调用需重新设置集合,效率随数量增长下降。 - 需遍历所有描述符检查状态,时间复杂度$O(n)$。 --- #### 五、典型应用场景(TCP服务器) ```c // 伪代码示例 int main() { int listen_fd = socket(...); bind(listen_fd, ...); listen(listen_fd, ...); fd_set read_fds; while(1) { FD_ZERO(&read_fds); FD_SET(listen_fd, &read_fds); // 监听新连接 // 添加已连接客户端的fd到read_fds select(max_fd + 1, &read_fds, NULL, NULL, NULL); if (FD_ISSET(listen_fd, &read_fds)) { // 接受新连接 int client_fd = accept(...); FD_SET(client_fd, &read_fds); } // 遍历检查所有客户端fd是否可读 for (each client_fd in read_fds) { if (FD_ISSET(client_fd, &read_fds)) { recv(...); // 读取数据 } } } } ``` --- #### 六、注意事项 1. **文件描述符上限** 通过宏 `FD_SETSIZE` 定义(通常1024),需重新编译内核才能修改。 2. **每次调用后集合会被修改** 需在循环中重新设置集合。 3. **正确处理异常事件** 如TCP带外数据(OOB)或连接错误。 4. **性能瓶颈** 高并发场景建议改用 `epoll`(Linux)或 `kqueue`(BSD)。 --- #### 七、与其他技术对比 | 特性 | select | poll | epoll | |-------------|--------------|--------------|--------------| | 最大描述符数量 | 1024 | 无限制 | 无限制 | | 时间复杂度 | $O(n)$ | $O(n)$ | $O(1)$ | | 触发模式 | 水平触发 | 水平触发 | 支持边缘触发 | | 内存操作 | 每次拷贝全量集合 | 每次拷贝全量集合 | 内核维护红黑树 | --- 通过 `select` 可简化多I/O场景的代码逻辑,但其性能限制使其更适用于中小规模并发场景。理解其原理和限制后,可根据实际需求选择更高效的替代方案。
评论 105
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

猿力猪

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值