Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >【面试题精讲】javaIO模型之NIO

【面试题精讲】javaIO模型之NIO

作者头像
程序员朱永胜
发布于 2023-10-22 00:45:17
发布于 2023-10-22 00:45:17
22200
代码可运行
举报
运行总次数:0
代码可运行

1. 什么是NIO?

NIO(New I/O)是Java提供的一种非阻塞I/O模型,它在JDK 1.4中引入。与传统的I/O模型相比,NIO提供了更高效、更灵活的I/O操作方式。

2. 为什么需要NIO?

传统的I/O模型使用阻塞式I/O,在进行读写操作时会导致线程被阻塞,直到数据准备好或者写入完成。这种模型对于并发处理能力较弱,当有大量连接同时请求时,每个连接都需要一个独立的线程来处理,造成资源浪费和性能下降。

而NIO采用了事件驱动的方式,通过Selector轮询注册的通道,只有在通道真正有读写事件发生时才会进行处理,避免了线程被阻塞的情况,提高了系统的并发处理能力。

3. NIO的实现原理?

NIO的核心组件包括:Channel、Buffer和Selector。

  • Channel: 类似于传统I/O中的流,可以通过Channel进行数据的读取和写入。常见的Channel类型有SocketChannel、ServerSocketChannel、FileChannel等。
  • Buffer: 缓冲区,用于存储数据。在NIO中,所有的数据都是通过Buffer进行读写的。常见的Buffer类型有ByteBuffer、CharBuffer、IntBuffer等。
  • Selector: 选择器,用于监听多个Channel的事件。通过Selector可以实现单线程处理多个通道的读写操作。

NIO的工作原理如下:

  1. 创建一个Selector,并将其注册到需要监听的Channel上。
  2. 当有数据准备好时,Selector会轮询已注册的Channel,发现有事件发生的Channel后进行处理。
  3. 根据不同的事件类型(读、写、连接、接收),使用对应的方法进行处理。

4. NIO的使用示例

以下是一个简单的NIO服务器示例,用于接收客户端发送的消息并返回相同的消息给客户端:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator;

public class NIOServer {
    public static void main(String[] args) throws IOException {
        // 创建ServerSocketChannel
        ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
        serverSocketChannel.socket().bind(new InetSocketAddress(8888));
        serverSocketChannel.configureBlocking(false);

        // 创建Selector
        Selector selector = Selector.open();
        serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);

        while (true) {
            // 阻塞等待就绪的Channel
            int readyChannels = selector.select();

            if (readyChannels == 0) {
                continue;
            }

            // 获取就绪的SelectionKey集合
            Iterator<SelectionKey> keyIterator = selector.selectedKeys().iterator();

            while (keyIterator.hasNext()) {
                SelectionKey key = keyIterator.next();

                if (key.isAcceptable()) {
                    // 接收新的连接
                    ServerSocketChannel serverChannel = (ServerSocketChannel) key.channel();
                    SocketChannel socketChannel = serverChannel.accept();
                    socketChannel.configureBlocking(false);
                    socketChannel.register(selector, SelectionKey.OP_READ);
                } else if (key.isReadable()) {
                    // 读取数据
                    SocketChannel socketChannel = (SocketChannel) key.channel();
                    ByteBuffer buffer = ByteBuffer.allocate(1024);
                    int bytesRead = socketChannel.read(buffer);

                    if (bytesRead > 0) {
                        buffer.flip();
                        byte[] data = new byte[buffer.remaining()];
                        buffer.get(data);
                        String message = new String(data, "UTF-8");
                        System.out.println("Received message: " + message);

                        // 返回相同的消息给客户端
                        ByteBuffer responseBuffer = ByteBuffer.wrap(data);
                        socketChannel.write(responseBuffer);
                    } else if (bytesRead < 0) {
                        // 客户端关闭连接
                        socketChannel.close();
                    }
                }

                // 移除处理过的SelectionKey
                keyIterator.remove();
            }
        }
    }
}

5. NIO的优点

  • 高并发性:NIO采用了事件驱动的方式,可以使用单线程处理多个通道的读写操作,提高系统的并发处理能力。
  • 非阻塞式I/O:NIO模型中的通道是非阻塞的,不会因为某个通道的读写操作而导致线程被阻塞,提高了系统的响应速度和吞吐量。
  • 内存管理优化:NIO使用了直接内存缓冲区,可以减少数据在Java堆和操作系统之间的拷贝次数,提高了I/O性能。

6. NIO的缺点

  • 复杂性较高:相比传统的阻塞式I/O模型,NIO的编程模型更加复杂,需要处理事件驱动、多路复用等概念。
  • 对编程人员要求较高:由于NIO的复杂性,对编程人员的技术要求较高,需要熟悉NIO相关的API和底层原理。

7. NIO的使用注意事项

  • 需要合理设置缓冲区大小:过小的缓冲区可能导致频繁的读写操作,影响性能;过大的缓冲区可能造成资源浪费。
  • 注意正确释放资源:在使用完Channel和Buffer后,需要及时关闭并释放资源,避免出现资源泄露问题。
  • 谨慎处理异常情况:NIO中的异常处理相对复杂,需要仔细处理各种异常情况,以保证程序的稳定性和可靠性。

8. 总结

NIO是一种非阻塞I/O模型,通过事件驱动和选择器机制实现高效的I/O操作。它具有高并发性、非阻塞式I/O和内存管理优化等优点,但也存在复杂性较高和对编程人员要求较高的缺点。在使用NIO时需要注意合理设置缓冲区大小、正确释放资源和谨慎处理异常情况。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023-10-17,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
走进Java NIO的奇妙世界:解锁高效IO操作的魔法
欢迎来到Java NIO的神奇之旅!在这个充满活力的世界里,我们将一起揭示Java NIO(New I/O)的奥秘,探索其在高效IO操作中的神奇魔法。无需担心,即使你是Java的小白,也能轻松领略这个强大而灵活的IO框架的魅力。
繁依Fanyi
2024/01/23
2370
BIO、NIO、IO多路复用模型的演进&Java NIO 网络编程
上文介绍了网络编程的基础知识,并基于 Java 编写了 BIO 的网络编程。我们知道 BIO 模型是存在巨大问题的,比如 C10K 问题,其本质就是因其阻塞原因,导致如果想要承受更多的请求就必须有足够多的线程,但是足够多的线程会带来内存占用问题、CPU上下文切换带来的性能问题,从而造成服务端崩溃的现象。怎么解决这一问题呢?优化呗,所以后面就有了NIO、AIO、IO多路复用。本文将对这几个模型详细说明并基于 Java 编写 NIO。
王二蛋
2024/01/18
7800
Java IO 与 NIO:高效的输入输出操作探究
输入输出(IO)是任何编程语言中的核心概念,而在Java中,IO操作更是应用程序成功运行的基石。随着计算机系统变得越来越复杂,对IO的要求也日益增加。在本文中,我们将探讨Java IO和非阻塞IO(NIO)的重要性以及如何在Java中实现高效的输入输出操作。
程序那些事
2023/10/17
2590
探索Java通信面试的奥秘:揭秘IO模型、选择器和网络协议,了解面试中的必备知识点!
TCP(Transmission Control Protocol)是一种面向连接的可靠的传输协议。类似于打电话,它通过建立一个连接和保证数据的可靠传输来提高通信的可靠性。然而,由于要确保数据的可靠性,TCP协议会增加网络负担,效率相对较低。
努力的小雨
2023/11/16
2070
【Netty】NIO 选择器 ( Selector ) 通道 ( Channel ) 缓冲区 ( Buffer ) 网络通信案例
NIO 网络通信 服务器端 操作流程 , 与 BIO 原理类似 , 基本流程是 启动服务器套接字通道 , 创建选择器 , 将服务器套接字通道注册给选择器 , 监听客户端连接事件 , 客户端连接成功后 , 创建套接字通道 , 将新创建的通道注册给选择器 , 然后监听该通道的读取事件 ;
韩曙亮
2023/03/27
7320
【Netty】NIO 选择器 ( Selector ) 通道 ( Channel ) 缓冲区 ( Buffer ) 网络通信案例
深入探讨I/O模型:Java中的阻塞和非阻塞和其他高级IO应用
I/O(Input/Output)模型是计算机科学中的一个关键概念,它涉及到如何进行输入和输出操作,而这在计算机应用中是不可或缺的一部分。在不同的应用场景下,选择正确的I/O模型是至关重要的,因为它会影响到应用程序的性能和响应性。本文将深入探讨四种主要I/O模型:阻塞,非阻塞,多路复用,signal driven I/O,异步IO,以及它们的应用。
程序那些事
2023/10/20
2120
深度解析IO模型与Reactor模型:NIO、BIO、AIO对比与实践
在软件开发领域,IO(Input/Output)模型是至关重要的概念之一。它决定了程序如何与外部世界进行数据交互,直接影响到系统性能和并发处理能力。本篇博客将深入探讨常见的IO模型,特别是NIO(New I/O)与传统的BIO(Blocking I/O)和AIO(Asynchronous I/O)的区别,以及Reactor模型在IO处理中的应用。
疯狂的KK
2023/09/26
6250
深度解析IO模型与Reactor模型:NIO、BIO、AIO对比与实践
JavaIO流:NIO梳理
NIO 也叫 Non-Blocking IO 是同步非阻塞的 IO 模型。线程发起 IO 请求后,立即返回。同步指的是必须等待 IO 缓冲区内的数据就绪,而非阻塞指的是,用户线程不原地等待 IO 缓冲区,可以先做一些其他操作,但是要定时轮询检查 IO 缓冲区数据是否就绪。
栗筝i
2022/12/02
3280
JavaIO流:NIO梳理
java的NIO编程
Java NIO(New I/O)是一种可伸缩且非阻塞的 I/O,与传统的 Java IO 不同。它允许同时处理多个连接,并且能够更快地处理 I/O 操作,是构建高性能、可伸缩和并发应用程序的重要组成部分。
堕落飞鸟
2023/04/02
4590
终结全网!手写Netty面试题答案
创建一个线程,注册到 Selector,将 serversocketchannel 注册到Selector selectionKey 里就有具体的事件
JavaEdge
2021/10/18
2260
面试官:Java Nio的优缺点?可能的瓶颈有哪些?
NIO 提供了与传统 BIO 模型中的 Socket 和 ServerSocket 相对应的 SocketChannel 和 ServerSocketChannel 两种不同的套接字通道实现。
Java宝典
2023/08/30
6890
面试官:Java Nio的优缺点?可能的瓶颈有哪些?
NIO简述
一个 Buffer 本质上是内存中的一块,我们可以将数据写入这块内存,之后从这块内存获取数据
leobhao
2022/06/28
3130
NIO简述
NIO面试题
摘自【https://mp.weixin.qq.com/s/YIcXaH7AWLJbPjnTUwnlyQ】
红目香薰
2022/11/28
2000
『互联网架构』软件架构-io与nio线程模型reactor模型(上)(53)
PS:NIO不需要的代码里面根本没有多线程,实际上nio只有一个工作线程,一个线程可以为多个客人服务。
IT架构圈
2019/05/30
5440
Java NIO、Channel、Selector 详解
Buffer 是一个特定原始类型的容器。Buffer 是一个原始类型的线性的、有限序列,除了 Buffer 存储的内容外,关键属性还包括:capacity, limit 和 position。
Yano_nankai
2019/11/10
1.2K0
Java NIO、Channel、Selector 详解
NIO (New I/O)
NIO是一种同步非阻塞的I/O模型,在Java 1.4 中引入了 NIO 框架,对应 java.nio 包,提供了 Channel , Selector,Buffer等抽象。
崔笑颜
2020/06/08
8110
Java面试必问通信框架NIO,原理详解
新的输入/输出 (NIO) 库是在 JDK 1.4 中引入的,弥补了原来的 I/O 的不足,提供了高速的、面向块的 I/O。
李红
2019/05/31
1.3K0
Java面试必问通信框架NIO,原理详解
彻底搞懂NIO效率高的原理
这篇文章读不懂的没关系,可以先收藏一下。笔者准备介绍完epoll和NIO等知识点,然后写一篇Java网络IO模型的介绍,这样可以使Java网络IO的知识体系更加地完整和严谨。初学者也可以等看完IO模型介绍的博客之后,再回头看这些博客,会更加有收获。
全菜工程师小辉
2019/08/16
2.6K0
Java NIO?看这一篇就够了!
✎前言 现在使用NIO的场景越来越多,很多网上的技术框架或多或少的使用NIO技术,譬如Tomcat,Jetty。学习和掌握NIO技术已经不是一个JAVA攻城狮的加分技能,而是一个必备技能。在前面2篇文
方志朋
2019/06/21
1.1K0
Java NIO?看这一篇就够了!
深入探索Java BIO与NIO输入输出模型:基于文件复制和socket通信
Java BIO是一种同步阻塞的I/O模型,它是Java最早提供的I/O模型。在进行读写操作的时候,若使用BIO进行通信,则操作不再受到操作系统的控制,而是由应用程序自己控制。在BIO中,数据的读取写入必须阻塞在一个线程内等待其完成。
公众号:码到三十五
2024/03/19
1860
深入探索Java BIO与NIO输入输出模型:基于文件复制和socket通信
推荐阅读
相关推荐
走进Java NIO的奇妙世界:解锁高效IO操作的魔法
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验