IO 和NIO有什么区别?

devtools/2025/3/1 17:03:53/

IO 与 NIO 的区别详解

Java 中的 IO(Input/Output) 和 NIO(New IO 或 Non-blocking IO) 是两种不同的输入输出处理机制,主要区别体现在设计模型、性能优化和应用场景上。以下是详细对比:


1. 阻塞与非阻塞模型
  • 传统 IO阻塞式。线程调用 read() 或 write() 时会被阻塞,直到数据完成读写,期间线程无法执行其他操作。

    • 示例:服务端使用 ServerSocket 的 accept() 会阻塞线程,直到客户端连接。

  • NIO非阻塞式。线程通过通道(Channel)发起读写请求后,可立即返回结果(如 SocketChannel.read() 返回读取的字节数或 0),若数据未就绪,线程可处理其他任务。

    • 适用场景:高并发服务器(如聊天室),单线程可管理多个连接。


2. 数据处理方式
  • IO:基于流(Stream),单向流动(如 InputStream 只能读,OutputStream 只能写),逐字节处理数据,灵活性较低。

  • NIO:基于通道(Channel)和缓冲区(Buffer),数据需先读入缓冲区再处理,支持双向读写(如 FileChannel 可同时读写)。

    • 缓冲区操作:通过 flip() 切换读写模式,rewind() 重读数据,clear() 清空缓冲区。

    • 示例:读取文件时,数据从 FileChannel 写入 ByteBuffer,处理后从 ByteBuffer 写回通道。


3. 多路复用机制(Selector)
  • NIO 使用 Selector(选择器) 实现单线程管理多个通道。Selector 通过轮询检测哪些通道已就绪(如可读、可写),大幅减少线程资源消耗。

    • 示例:Web 服务器通过一个线程处理数千个客户端连接。

  • IO 每个连接需独立线程处理,线程数过多时会导致上下文切换开销大,甚至内存溢出。


4. 性能与适用场景
  • IO:适合低并发、数据量大的场景(如文件传输),代码简单。

  • NIO:适合高并发、小数据量的场景(如即时通讯),通过非阻塞和选择器提升吞吐量。

    • 零拷贝技术FileChannel.transferTo() 直接将数据从磁盘发送到网络,减少 CPU 拷贝次数。

    • 内存映射文件FileChannel.map() 将文件映射到内存,直接操作内存数据提升读写效率。


5. API 复杂度
  • IO:API 简单直观,如 FileInputStream/FileOutputStream

  • NIO:API 更复杂,需管理缓冲区、通道、选择器,开发难度较高。

    • 需处理边界条件(如缓冲区未读完的数据)、正确调用 flip()/clear() 等。


对比总结表

特性IONIO
阻塞模型阻塞式非阻塞式
数据结构流(单向)通道(双向) + 缓冲区
多路复用不支持支持(Selector 管理多个 Channel)
线程模型一连接一线程单线程处理多连接
适用场景低并发、大数据量(如文件传输)高并发、小数据量(如即时通讯服务器
高级特性内存映射文件、零拷贝

代码示例

传统 IO 读取文件:

try (FileInputStream fis = new FileInputStream("file.txt")) {int data;while ((data = fis.read()) != -1) {System.out.print((char) data);}
}

NIO 读取文件:

try (FileChannel channel = FileChannel.open(Paths.get("file.txt"))) {ByteBuffer buffer = ByteBuffer.allocate(1024);while (channel.read(buffer) > 0) {buffer.flip();  // 切换为读模式while (buffer.hasRemaining()) {System.out.print((char) buffer.get());}buffer.clear(); // 清空缓冲区,准备下一次写入}
}

总结

  • IO 适合简单的阻塞式任务,开发简单但资源消耗高。

  • NIO 在需要高并发和非阻塞处理时优势明显,但需处理更复杂的 API 和边界条件。选择时需根据具体场景权衡。


http://www.ppmy.cn/devtools/163679.html

相关文章

依赖注入

props逐级透析: provide在祖宗这里写这个方法: 父亲没动 直接在子里写inject即可: 全局数据;

【蓝桥杯】每天一题,理解逻辑(1/90)【Leetcode 移动零】

文章目录 题目解析讲解算法原理【双指针算法思路】(数组下标充当指针)如何划分和执行过程大致 代码详情 题目解析 题目链接:https://leetcode.cn/problems/move-zeroes/description/ 题目意思解析 把所有的零移动到数组的末尾保持非零元素的相对顺序 理解了这两层…

面试之《react hooks在源码中是怎么实现的?》

要深入理解 React Hooks 在源码中的实现,可以从以下几个关键方面来剖析: 核心数据结构 在 React 内部,使用链表来管理每个函数组件的 Hooks。每个 Hook 对应一个节点,这些节点通过 next 指针相连。以下是简化后的 Hook 节点结构…

《A++ 敏捷开发》- 17 持续集成

为了避免客户验收前或使用后才暴露大量棘手缺陷,可能要花很长时间才能发现并解决,便应依据精益和系统工程的原则,把系统拆分成子系统/模块,先开发并测试子系统/模块、集成、再测试,按部就班地完成整个软件开发。 验收…

Java中常见的设计模式

设计模式是软件设计中针对常见问题的可复用解决方案,它们提供了代码组织和架构的最佳实践,Java中常见的设计模式可分为创建型、结构型和行为型三类。下面就给大家介绍一些常用的设计模式和案例。 创建型模式:管理对象创建 1.单例模式 确保…

笔记二:整数和浮点数在内存中存储

目录 一、数据类型介绍 二、类型的基本归类 1.整形家族: 2.浮点数家族: 3.构造类型: 4.指针类型 5.空类型: 三、整形在内存中的存储 3.1 原码,反码、补码 3.2 大小端介绍 四、浮点数在内存中的存储 ​编辑 4.…

C++二叉搜索树查找,插入,删除

二叉搜索树(Binary Search Tree, BST)是一种二叉树,每个节点包含一个键值,并且具有以下特性: 左子树的所有节点的键值都小于当前节点的键值。右子树的所有节点的键值都大于当前节点的键值。每个节点的左右子树都是二叉…

【详细讲解在STM32的UART通信中使用DMA机制】

详细讲解在STM32的UART通信中使用DMA机制 目录 详细讲解在STM32的UART通信中使用DMA机制一、DMA机制概述二、DMA在UART中的作用三、DMA的配置步骤四、UART初始化与DMA结合五、DMA传输的中断处理六、DMA与中断的结合使用七、注意事项与常见问题八、代码示例九、总结 一、DMA机制…