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中String的intern()方法详细说明

    Java中String的intern()方法详细说明

    这篇文章主要介绍了Java中String的intern()方法详细说明,String::intern()是一个本地方法,他的作用就是如果字符串常量池中已经包含了一个等于此String对象的字符串,则返回代表池中的这个字符串额String对象的引用,需要的朋友可以参考下
    2023-11-11
  • java JOptionPane类的介绍

    java JOptionPane类的介绍

    java JOptionPane类的介绍,需要的朋友可以参考一下
    2013-04-04
  • 详述IntelliJ IDEA远程调试Tomcat的方法(图文)

    详述IntelliJ IDEA远程调试Tomcat的方法(图文)

    本篇文章主要介绍了详述IntelliJ IDEA远程调试Tomcat的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-12-12
  • 基于Java 注解(Annotation)的基本概念详解

    基于Java 注解(Annotation)的基本概念详解

    基于Java 注解(Annotation)的基本概念详解
    2013-04-04
  • Java详细讲解包的作用以及修饰符的介绍

    Java详细讲解包的作用以及修饰符的介绍

    本文主要讲述的是包的使用和注意事项和四种访问修饰符public,protected,默认的,private的访问范围及实例,感兴趣的朋友一起来看看
    2022-05-05
  • 深入理解java动态代理机制

    深入理解java动态代理机制

    本篇文章主要介绍了深入理解java动态代理机制,详细的介绍动态代理有哪些应用场景,什么是动态代理,怎样使用,它的局限性在什么地方?有兴趣的可以了解一下。
    2017-02-02
  • 如何批量测试Mybatis项目中的Sql是否正确详解

    如何批量测试Mybatis项目中的Sql是否正确详解

    这篇文章主要给大家介绍了关于如何批量测试Mybatis项目中Sql是否正确的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-12-12
  • Java Web导出等比例图片到Excel的实现过程

    Java Web导出等比例图片到Excel的实现过程

    我们使用Java导出图片到Excel,打开成功导出的Excel一看,商品对应的图片都很规矩的按照我的设置铺满了整个单元格,但是,商品图片却都变形了,这样肯定是不行的,于是第一反应就是将图片等比例导出,所以本文本给大家介绍了如何使用Java Web导出等比例图片到Excel
    2023-11-11
  • 学会Pulsar Consumer的使用方式

    学会Pulsar Consumer的使用方式

    这篇文章主要介绍了Pulsar Consumer的使用方式,全文使用大量的代码来做了详细的讲解,感兴趣的小伙伴可以参考一下这篇文章,希望读完能对你有很大的帮助
    2021-08-08
  • 浅析java中String类型中“==”与“equal”的区别

    浅析java中String类型中“==”与“equal”的区别

    这篇文章主要介绍了浅析java中String类型中“==”与“equal”的区别,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08

最新评论