一、基本概念

Java NIO 是 Java 1.4 引入的,用于处理高速、高并发的 I/O 操作。与传统的阻塞 I/O 不同,NIO 支持非阻塞 I/O 和选择器,可以更高效地管理多个通道。

二、核心组件

  1. 通道(Channel)

    • Channel 是 NIO 中用于读取和写入数据的主要接口,提供双向数据传输的能力。
    • 常见的通道实现:
      • FileChannel:用于文件的读写操作。
      • SocketChannel:用于 TCP 网络通信。
      • ServerSocketChannel:用于监听 TCP 连接的服务器端通道。
      • DatagramChannel:用于 UDP 网络通信。
  2. 缓冲区(Buffer)
    • Buffer 是 NIO 中用于存储数据的容器。与传统的流不同,NIO 通过缓冲区进行数据的读写。
    • 常见的缓冲区类型:
      • ByteBuffer:处理字节数据。
      • CharBuffer:处理字符数据。
      • IntBufferLongBuffer 等:处理整型和长整型数据。
    • 缓冲区有三个重要的属性:
      • position:当前缓冲区的读写位置。
      • limit:可以读取或写入的最大数据量。
      • capacity:缓冲区的总容量。
  3. 选择器(Selector)
    • Selector 是 NIO 的核心组件之一,允许单个线程监控多个通道的事件。
    • 通过选择器,可以处理多个连接而不需要为每个连接都创建一个线程。
    • Selector 的工作流程:
      • 注册通道(Channel)到选择器。
      • 选择感兴趣的通道(如可读、可写、连接等)。
      • 处理就绪的通道。

三、底层实现

  1. 文件描述符

    NIO 底层仍然依赖操作系统的文件描述符。每个通道对应一个文件描述符,用于直接与操作系统进行交互。

  2. 事件驱动

    NIO 使用事件驱动的机制。选择器会调用操作系统的底层 API(如 epoll、kqueue)来获取就绪事件。这种机制允许线程在等待事件时处于睡眠状态,从而减少 CPU 资源的消耗。

四、设计原理

  1. 非阻塞 IO

    NIO 允许通道在没有可用数据时不阻塞线程。线程可以继续执行其他操作,适合处理高并发请求。

  2. 选择性处理

    使用选择器,可以选择性地处理就绪通道,避免了为每个连接创建一个线程的开销。

  3. 适应性强

    NIO 的设计使得它可以处理各种数据源(如文件、网络等),提高了灵活性。

五、底层原理

  1. 内存管理

    NIO 的缓冲区(Buffer)底层使用 java.nio.HeapByteBufferjava.nio.DirectByteBuffer,后者直接在 JVM 之外分配内存,减少了与 JVM 堆内存的交互开销,提升了 I/O 性能,特别是在大数据量传输时。

  2. 内存映射文件(Memory-Mapped File)

    NIO 的 FileChannel 支持内存映射文件,允许将文件映射到内存。这种方式使得文件内容可以像数组一样直接操作,大幅提升了文件读取和写入的速度,特别适用于大文件处理和高性能数据库实现。

  3. 选择器的实现

    选择器的实现通常基于操作系统提供的高效 I/O 多路复用机制,如 Linux 的 epoll 或 Windows 的 IOCP。这些机制使得 NIO 能够在处理大量并发连接时表现优异。了解这些底层实现的机制,能够帮助开发者在不同操作系统上优化性能。

六、使用场景

  • 高性能 Web 服务器

    NIO 适合构建高性能的 Web 服务器,如 Netty 框架,利用其事件驱动和异步非阻塞的特性,可以处理数万并发连接,而不需要为每个连接创建一个线程。

  • 实时数据处理

    在需要实时处理大量数据的应用(如金融交易系统、在线游戏等),NIO 提供的低延迟和高吞吐量使其成为理想选择。

  • 跨平台的网络通信

    NIO 的通道和选择器机制提供了跨平台的网络通信能力,开发者可以轻松构建支持多种操作系统的网络应用。

  • 高并发网络应用

    NIO 适用于需要处理大量并发连接的应用,例如聊天服务器、HTTP 服务器和在线游戏等。

  • 异步文件处理

    使用 AsynchronousFileChannel 进行异步文件读写操作,适合需要高性能的文件处理场景。

七、性能特点

  1. 降低上下文切换

    NIO 的非阻塞特性降低了线程切换的开销,特别是在高并发情况下,提高了应用的吞吐量。

  2. 内存映射文件

    NIO 支持内存映射文件,可以将文件直接映射到内存,这种方式可以提高对大文件的访问速度。

  3. 减少资源占用

    由于使用选择器管理多个通道,NIO 可以减少对系统资源(如线程和内存)的占用,提高整体性能。

八、示例代码

以下是一个简单的 NIO 服务器示例,使用选择器处理客户端连接:

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.nio.channels.SelectionKey; public class NioServer {
public static void main(String[] args) throws IOException {
Selector selector = Selector.open();
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.bind(new InetSocketAddress(8080));
serverSocketChannel.configureBlocking(false);
serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT); while (true) {
selector.select(); // 阻塞,直到有事件发生
for (SelectionKey key : selector.selectedKeys()) {
if (key.isAcceptable()) {
SocketChannel socketChannel = serverSocketChannel.accept();
socketChannel.configureBlocking(false);
socketChannel.register(selector, SelectionKey.OP_READ);
} else if (key.isReadable()) {
SocketChannel socketChannel = (SocketChannel) key.channel();
ByteBuffer buffer = ByteBuffer.allocate(256);
int bytesRead = socketChannel.read(buffer);
if (bytesRead == -1) {
socketChannel.close();
} else {
buffer.flip();
// 处理数据...
socketChannel.write(buffer);
}
}
}
selector.selectedKeys().clear(); // 清除已处理的事件
}
}
}

一文彻底熟练掌握并使用Java的NIO操作的更多相关文章

  1. 使用DOM进行xml文档的crud(增删改查)操作<操作详解>

    很多朋友对DOM有感冒,这里我花了一些时间写了一个小小的教程,这个能看懂,会操作了,我相信基于DOM的其它API(如JDOM,DOM4J等)一般不会有什么问题. 后附java代码,也可以下载(可点击这 ...

  2. java之NIO编程

    所谓行文如编程,随笔好比java文件,文章好比类,参考文献是import,那么目录就是方法定义. 本篇文章处在分析thrift的nonblocking server之前,因为后者要依赖该篇文章的知识. ...

  3. JAVA 探究NIO

    事情的开始 1.4版本开始,java提供了另一套IO系统,称为NIO,(New I/O的意思),NIO支持面向缓冲区的.基于通道的IO操作. 1.7版本的时候,java对NIO系统进行了极大的扩展,增 ...

  4. java字符流操作flush()方法及其注意事项

    java字符流操作flush()方法及其注意事项   flush()方法介绍 查阅文档可以发现,IO流中每一个类都实现了Closeable接口,它们进行资源操作之后都需要执行close()方法将流关闭 ...

  5. java 的nio与io对比

    转:本文并非Java.io或Java.nio的使用手册,也不是如何使用Java.io与Java.nio的技术文档.这里只是尝试比较这两个包,用最简单的方式突出它们的区别和各自的特性.Java.nio提 ...

  6. java使用POI操作XWPFDocument中的XWPFRun(文本)对象的属性详解

    java使用POI操作XWPFDocument中的XWPFRun(文本)对象的属性详解 我用的是office word 2016版 XWPFRun是XWPFDocument中的一段文本对象(就是一段文 ...

  7. Tinking in Java ---Java的NIO和对象序列化

    前面一篇博客的IO被称为经典IO,因为他们大多数都是从Java1.0开始就有了的:然后今天这篇博客是关于NIO的,所以的NIO其实就是JDK从1.4开始,Java提供的一系列改进的输入/输出处理的新功 ...

  8. 少啰嗦!一分钟带你读懂Java的NIO和经典IO的区别

    1.引言 很多初涉网络编程的程序员,在研究Java NIO(即异步IO)和经典IO(也就是常说的阻塞式IO)的API时,很快就会发现一个问题:我什么时候应该使用经典IO,什么时候应该使用NIO? 在本 ...

  9. Java BIO NIO 与 AIO

    回顾 上一章我们介绍了操作系统层面的 IO 模型. 阻塞 IO 模型. 非阻塞 IO 模型. IO 复用模型. 信号驱动 IO 模型(用的不多,知道个概念就行). 异步 IO 模型. 并且介绍了 IO ...

  10. Java 正则表达式实例操作

    Regular Expression正则表达式,简称RegExp,常规通用的表达式,在多个开发语言中都有它的实现,可以通过正则表达式来快速的检索.匹配.查找.替换字符串中的文本. 简单实例 匹配网址 ...

随机推荐

  1. 2024-08-14:用go语言,给定两个长度分别为n和m的整数数组nums和changeIndices,下标从1开始。初始时,nums 中所有下标均未标记。 从第1秒到第m秒,每秒可以选择以下四种操

    2024-08-14:用go语言,给定两个长度分别为n和m的整数数组nums和changeIndices,下标从1开始.初始时,nums 中所有下标均未标记. 从第1秒到第m秒,每秒可以选择以下四种操 ...

  2. 牛客周赛 Round 6

    牛客周赛 Round 6 A-游游的数字圈_牛客周赛 Round 6 (nowcoder.com) 枚举即可 #include <bits/stdc++.h> #define int lo ...

  3. 我是如何使用 vue2+element-ui 处理负责表单,避免单文件过大的问题

    引言 在工作中我经常需要处理一些复杂.动态表单,但是随着需求不断迭代,我们也许会发现曾经两三百行的.vue文件现在不知不觉到了两千行,三千行,甚至更多... 这对于一个需要长期维护的项目,无疑是增加了 ...

  4. WM_LBUTTONDOWN,WM_LBUTTONUP

    WM_LBUTTONDOWN //鼠标左键按下消息WM_LBUTTONUP //鼠标左键弹起消息参数和按下一样 当用户在窗口的客户区域中按住鼠标左键时,会发布WM_LBUTTONDOWN消息.如果未捕 ...

  5. Win32 插入符光标跟随的打字小程序

    1.先创建插入符光标 在WM_CREATE消息中 LRESULT OnCreate(HWND hWnd, WPARAM wParam, LPARAM lParam) { HDC hdc = GetDC ...

  6. 组合数取模的几种方法--Exlucas&杨辉三角&组合

    组合数取模的几个方法 求: \[C^{m}_{n} \bmod P \] 1.杨辉三角法 \[C^{m}_{n} = C^{m - 1}_{n - 1} + C^{ m }_{n - 1} \] 时间 ...

  7. 禁止 SSH 传递 locale 环境变量

    SSH 在连接远程机器时默认会传递一些环境变量,其中就包括你本机的 locale 变量.这会导致远程机器的 locale 配置变成和你本地主机一样.有时候我们不希望这种行为,我们可以通过修改 SSH ...

  8. drawable xx should not reference itself

    背景: 在Android中新增一个xx.xml,在layer-list 的item中设定引入的drawable后,报这个提示(xx不能引用自身) 原因: 这个错误其实很离谱,但是还是有必要记一下,万一 ...

  9. c++字符编码转换

    c++字符编码转换 简述 字符编码一直是软件开发中很麻烦的问题.当前项目开发普遍使用的字符集是utf-8,而windows系统则默认是gbk,linux默认编码则是utf-8,所以想要开发一个在win ...

  10. Hydra(海德拉)工具使用从0到1,爆破服务器密码,2024最新版

    Hydra(海德拉)工具使用从0到1,爆破服务器密码,2024最新版 Hydra简介 Hydra又叫九头蛇,是一款由著名的黑客组织THC开发的开源暴力破解工具,支持大部分协议的在线密码破解,是网络安全 ...