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

IO模型IO模型

一、基础概念:什么是IO模型?

  • IO模型实际上描述的是操作系统(或程序)和设备(如网络、硬盘等)之间数据传输的方式和策略。
  • 在网络编程里,常常涉及等待数据的到来,处理请求的效率是关键。不同的IO模型决定了程序在等待数据时的表现。

二、五种IO模型详解

1. 阻塞IO(Blocking IO)

原理:

  • 程序发起IO请求后(如read()、recv()),如果没有数据到达,则程序会“阻塞”在这里,等待数据到来。
  • 直到数据准备好,函数调用才返回。

图示:

程序调用read() -> 阻塞等待数据 -> 数据到来 -> 读取完成 -> 继续执行

优点:

  • 实现简单,使用直观。
  • 传统经典模型。

缺点:

  • 阻塞期间不能做其他事情,效率低。
  • 一个连接阻塞,可能导致整个程序等待。

适用场景:

  • 小型应用,连接少,简单实现。

2. 非阻塞IO(Non-Blocking IO)

原理:

  • 设置文件描述符(socket)为非阻塞模式,调用read()时立即返回。
  • 如果没有数据到来,read()会立即返回错误(一般是EAGAINEWOULDBLOCK)。
  • 程序需要不断地轮询(轮询就是不停地检查),或者结合其他机制。

图示:

程序调用read() -> 如果没有数据,立即返回错误 -> 继续做其他事,之后再次尝试

优点:

  • 不会被阻塞,可以同时发起多个请求。
  • 适合需要同时处理大量连接的场景。

缺点:

  • 需要不断“轮询”发现数据,浪费CPU资源。
  • 编程复杂度较高。

适用场景:

  • 需要高并发的网络服务器(但通常结合IO复用使用更常见)。

3. IO复用(IO Multiplexing)

原理:

  • 利用select()poll()epoll()等系统调用,监听多个文件描述符的“是否有事件发生”。
  • 让程序“等待”一组文件描述符的状态变化(如准备好读写),一旦某个就绪,通知程序处理。

示意图:

多个连接(描述符)都在监视中
调用select() -> 等待就绪的描述符 -> 处理就绪的连接

优点:

  • 可以同时监控大量连接。
  • 避免轮询带来的CPU浪费。

缺点:

  • select()poll()效率在大量连接时变差(因为每次都要扫描所有描述符)。
  • epoll()(Linux特有)优化更好,支持高效大量连接。

适用场景:

  • 高性能网络服务器(比如Web服务器)。

4. 信号驱动IO(Signal-Driven IO)

原理:

  • 通过注册信号(如SIGIO)通知程序,当设备(或连接)准备好数据时,系统给程序发信号。
  • 程序在接到信号后,执行相应的处理。

操作流程:

  1. 设置socket的异步通知(用fcntl()等)。
  2. 当数据到达,系统触发信号(SIGIO)。
  3. 信号处理函数被调用,程序可以读取数据。

优点:

  • 让程序在事件发生时得到通知,不用主动跑轮询。
  • 可以处理多个IO事件。

缺点:

  • 编程复杂,信号处理难以调试。
  • 信号可能丢失或难以同步。

适用场景:

  • 需要异步通知的特殊应用。

5. 异步IO(Asynchronous IO)

原理:

  • 发送异步IO请求后,调用会立即返回,数据在后台进行传输。
  • 通过回调函数或事件通知,程序得知数据已准备好。
  • Linux中实现可用aio_*接口或者现代的io_uring

示意图:

发请求 -> 立即返回 -> 后台进行操作 -> 数据准备好,通知应用

优点:

  • 非常高效,处理大量IO请求时性能出色。
  • 让程序可以继续做其他事,不用等待。

缺点:

  • 编程模型复杂(需要理解回调、事件等机制)。
  • 依赖操作系统支持(Linux的io_uring是最新的技术)。

适用场景:

  • 高性能、大规模的网络或硬盘IO。

三、小结:五种IO模型的对比

模型是否阻塞复杂度适用场景典型使用方法
阻塞IO阻塞简单小规模应用,简单场景直接调用read()/write()
非阻塞IO不阻塞中等高并发、轮询场景设置为非阻塞+轮询
IO复用不阻塞中等同时管理大量连接select()/poll()/epoll()
信号驱动IO不阻塞事件驱动少量连接,特殊应用注册信号处理
异步IO不阻塞高性能、特大量请求处理aio接口、io_uring

四、总结

  • 阻塞IO简单但效率低,适合小型应用。
  • 非阻塞IO允许程序自己轮询,适合短连接。
  • IO复用selectepoll监控多个连接,是网络服务器的基础。
  • 信号驱动异步IO更复杂,但效率更高,适合高性能场景。
http://www.xdnf.cn/news/771175.html

相关文章:

  • ISO18436-2 CATII级振动分析师能力矩阵
  • 振动分析师(ISO18436-2)四级能力矩阵 - 简介
  • ROS机器人和NPU的往事和新知-250602
  • python打卡训练营打卡记录day43
  • 阿里云服务器-解决宝塔登录不成功
  • DAY 41 简单CNN
  • 审计 - 风险应对 - 控制测试
  • springboot04
  • 2025年十大AI幻灯片工具深度评测与推荐
  • C++.cstring string
  • 线段树刷题记录
  • 数据治理的演变与AI趋势
  • 零基础开始的网工之路第十七天------计算机网络知识
  • 机器学习——集成学习
  • STM32入门教程——GPIO输入
  • 使用Mathematica观察多形式根的分布随参数的变化
  • mysql数据库实现分库分表,读写分离中间件sharding-sphere
  • 数据库MySQL集群MGR
  • NiceGUI 是一个基于 Python 的现代 Web 应用框架
  • PyTorch——卷积层(3)
  • MapReduce(期末速成版)
  • 检索器组件深入学习与使用技巧 VectorStoreRetriever 检索器
  • android binder(二)应用层编程实例
  • 基于 Android 和 JBox2D 的简单小游戏
  • 【短距离通信】【WiFi】精讲WLAN 驱动结构
  • Android Studio 之基础代码解析
  • wow Warlock shushia [Dreadsteed]
  • 【Java EE初阶 --- 多线程(初阶)】多线程的实现案例
  • 《Effective Python》第六章 推导式和生成器——使用 yield from 组合多个生成器
  • 嵌入式学习笔记 - FreeRTOS关于vApplicationGetIdleTaskMemory