Java中AIO、BIO、NIO应用场景及区别

 更新时间:2023年06月12日 10:35:48   作者:激流丶  
本文主要介绍了Java中AIO、BIO、NIO应用场景及区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、AIO、BIO、NIO 的区别

AIO(异步 I/O):AIO 是 Java NIO 2 中新增的一种 I/O 模式,它的特点是 I/O 操作不会阻塞线程,而是在后台由操作系统完成,完成后会通知应用程序。AIO 可以让应用程序在等待 I/O 完成时执行其他任务,提高了系统的并发性能。AIO 适用于高并发的网络应用,比如聊天室、多人在线游戏等。

BIO(阻塞 I/O):BIO 是 Java 中最早的一种 I/O 模式,它的特点是 I/O 操作会阻塞线程,直到 I/O 操作完成。BIO 的缺点是并发性能较差,因为每个线程都会阻塞等待 I/O 完成。BIO 适用于连接数较少的网络应用,比如 Web 应用中的 Servlet。

NIO(非阻塞 I/O):NIO 是 Java 中的一种 I/O 模式,它的特点是 I/O 操作不会阻塞线程,但是需要轮询操作系统的 I/O 事件来判断是否有 I/O 操作完成。NIO 可以让应用程序在等待 I/O 完成时执行其他任务,提高了系统的并发性能。NIO 适用于连接数较多、并发性要求较高的网络应用,比如高性能的服务器应用、网关应用等。

二、应用场景

AIO、BIO、NIO 适用于 Java 网络编程,可以用于开发各种网络应用。以下是它们的一些应用场景:

AIO:适用于高并发的网络应用,比如聊天室、多人在线游戏等。

BIO:适用于连接数较少的网络应用,比如 Web 应用中的 Servlet。

NIO:适用于连接数较多、并发性要求较高的网络应用,比如高性能的服务器应用、网关应用等。

三、NIO 的举例

假设有一个服务器需要同时处理多个客户端请求,传统的 BIO 模式需要为每个客户端请求创建一个线程,这样会导致线程数量过多,占用大量的系统资源。而 NIO 模式可以通过单线程处理多个客户端请求,极大地减少了线程数量,提高了系统的并发性能(详细可了解后面的IO多路复用)。

具体实现方法是,使用 NIO 的 Selector 对象来监听多个通道的事件,当某个通道有事件发生时,Selector 会通知应用程序处理该事件。下面是一个简单的 NIO 服务器实现代码:

package com.pany.camp.io;
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;
import java.util.Set;
public class NIOServer {
    public static void main(String[] args) throws IOException {
        // 创建一个 Selector 对象
        Selector selector = Selector.open();
        // 创建一个 ServerSocketChannel 对象,并绑定到指定端口
        ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
        serverSocketChannel.socket().bind(new InetSocketAddress(8080));
        serverSocketChannel.configureBlocking(false);
        // 将 ServerSocketChannel 注册到 Selector 中,监听 ACCEPT 事件
        serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
        while (true) {
            // 阻塞等待事件发生
            selector.select();
            // 处理事件
            Set<SelectionKey> selectionKeys = selector.selectedKeys();
            Iterator<SelectionKey> iterator = selectionKeys.iterator();
            while (iterator.hasNext()) {
                SelectionKey selectionKey = iterator.next();
                if (selectionKey.isAcceptable()) {
                    // 处理 ACCEPT 事件
                    ServerSocketChannel ssc = (ServerSocketChannel) selectionKey.channel();
                    SocketChannel socketChannel = ssc.accept();
                    socketChannel.configureBlocking(false);
                    socketChannel.register(selector, SelectionKey.OP_READ);
                } else if (selectionKey.isReadable()) {
                    // 处理 READ 事件
                    SocketChannel socketChannel = (SocketChannel) selectionKey.channel();
                    ByteBuffer buffer = ByteBuffer.allocate(1024);
                    socketChannel.read(buffer);
                    buffer.flip();
                    System.out.println("Received message: " + new String(buffer.array()));
                }
                // 处理完事件后,将该事件从 selectedKeys 集合中移除
                iterator.remove();
            }
        }
    }
}

以上代码创建了一个 NIO 服务器,使用单线程处理多个客户端请求。当有客户端连接时,会触发 ACCEPT 事件,将该客户端的 SocketChannel 注册到 Selector 中,监听 READ 事件。当客户端发送数据时,会触发 READ 事件,服务器可以读取客户端发送的数据并进行处理。

四、NIO 在 Netty 中的使用

Netty 是一个基于 NIO 的高性能网络编程框架,它封装了 NIO 的底层细节,提供了更加简单易用的 API,使得开发高性能的网络应用变得更加容易。下面是 Netty 中 NIO 的一些应用:

Channel:在 Netty 中,Channel 是 NIO 中的 SocketChannel 的封装,它提供了更加简单易用的 API,比如可以通过 ChannelPipeline 来实现数据的编解码、流量控制、拆包粘包等功能。

EventLoop:在 Netty 中,EventLoop 是 NIO 中的 Selector 的封装,它负责处理所有的 I/O 事件,包括连接、读、写等事件。通过 EventLoop 的多线程处理能力,可以实现高并发的网络应用。

ByteBuf:在 Netty 中,ByteBuf 是 NIO 中的 ByteBuffer 的封装,它提供了更加灵活的内存管理方式,可以实现零拷贝的数据传输。

ChannelFuture:在 Netty 中,ChannelFuture 是 NIO 中的 Future 的封装,它提供了更加简单易用的异步编程方式,可以实现非阻塞的网络应用。

ChannelHandlerContext:在 Netty 中,ChannelHandlerContext 是 ChannelPipeline 中的一个节点,它提供了更加灵活的事件处理方式,可以实现自定义的数据编解码、流量控制、拆包粘包等功能。

Netty 是一个基于 NIO 的高性能网络编程框架,它封装了 NIO 的底层细节,提供了更加简单易用的 API,使得开发高性能的网络应用变得更加容易。

到此这篇关于Java中AIO、BIO、NIO应用场景及区别的文章就介绍到这了,更多相关Java AIO BIO NIO内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java 20在Windows11系统下的简易安装教程

    Java 20在Windows11系统下的简易安装教程

    这篇文章主要给大家介绍了关于Java 20在Windows11系统下的简易安装教程,学习Java的同学,第一步就是安装好Java环境,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2023-07-07
  • Java与Unix时间戳的相互转换详解

    Java与Unix时间戳的相互转换详解

    这篇文章主要为大家详细介绍了Java与Unix时间戳的相互转换,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • kkFileView在线预览office的常见问题以及解决方案

    kkFileView在线预览office的常见问题以及解决方案

    kkFileView在线预览Office常见问题包括base64编码配置、Office组件安装、乱码处理及水印添加,解决方案涉及版本适配、安装LibreOffice、设置编码和字体路径,以及通过URL参数传递水印文本
    2025-07-07
  • 深入浅出Java 抽象类与接口及区别对比分析

    深入浅出Java 抽象类与接口及区别对比分析

    抽象类和接口是Java中实现多态和设计模式的重要工具,它们在编程中具有重要的地位,能够有效提高代码的灵活性和可维护性,本文给大家介绍Java抽象类与接口及区别对比分析,感兴趣的朋友跟随小编一起看看吧
    2025-11-11
  • Java中StringUtils与CollectionUtils和ObjectUtil概念讲解

    Java中StringUtils与CollectionUtils和ObjectUtil概念讲解

    这篇文章主要介绍了Java中StringUtils与CollectionUtils和ObjectUtil概念,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2022-12-12
  • Java加载ICC文件的方法和示例代码

    Java加载ICC文件的方法和示例代码

    ICC文件,通常用于颜色管理,定义了如何将一个颜色空间转换为另一个颜色空间,在Java中,我们可能需要加载这些文件来进行颜色转换或管理,本文将为您提供加载ICC文件的方法和示例代码,需要的朋友参考下吧
    2023-08-08
  • JavaScript不使用临时变量交换两个变量值的方法总结

    JavaScript不使用临时变量交换两个变量值的方法总结

    在 JavaScript 中交换两个变量的值是一项基础但重要的操作,传统方法会使用一个临时变量,但在某些情况下,我们可能需要不使用临时变量来实现交换,本文将详细介绍多种不使用临时变量交换两个变量值的方法,需要的朋友可以参考下
    2025-04-04
  • HTTP中get和post的区别详解

    HTTP中get和post的区别详解

    这篇文章主要为大家详细介绍了HTTP中get和post的区别,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-08-08
  • spring boot使用sonarqube来检查技术债务

    spring boot使用sonarqube来检查技术债务

    今天小编就为大家分享一篇关于spring boot使用sonarqube来检查技术债务,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • Java实现异步转同步的多种方法介绍

    Java实现异步转同步的多种方法介绍

    这篇文章主要为大家详细介绍了Java实现异步转同步的多种方法的实现与对比,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下
    2025-09-09

最新评论