当前位置: 首页 > news >正文

聊聊分布式架构05——[NIO基础]BIO到NIO的演进

目录

I/O

I/O模型

BIO示例

BIO与NIO比较

NIO的三大核心

NIO核心之缓冲区

Buffer常用子类:

Buffer常用API

Buffer中的重要概念

NIO核心之通道

FileChannel 类

FileChannel常用方法

NIO核心之选择器

概述

应用

NIO非阻塞原理分析

服务端流程

客户端流程

简单NIO示例

AIO简介

主要特点

主要组件和类

总结



I/O

I/O(Input/Output)是计算机科学和编程中的一个重要概念,指的是计算机程序与外部世界(通常是硬件设备、文件系统、网络、用户等)进行数据交换和通信的过程。I/O 可以包括从外部设备读取数据(输入)和将数据写入外部设备(输出)两个方面。以下是关于I/O的一些重要概念和说明:

  1. 输入(Input): 输入是指程序从外部获取数据的过程。这可以是从键盘、鼠标、传感器、文件、网络连接等获取数据。例如,用户通过键盘输入文本,程序从网络接收传感器数据,都属于输入操作。

  2. 输出(Output): 输出是指程序向外部设备发送数据的过程。这可以是将数据写入文件、向屏幕显示信息、通过网络传输数据等。例如,程序将计算结果写入文件,向用户显示图形界面,都属于输出操作。

  3. 流(Stream): 流是一种抽象概念,用于表示数据在程序和外部设备之间的有序、连续的传输。I/O流通常分为输入流和输出流。输入流用于从外部设备读取数据,输出流用于将数据发送到外部设备。

  4. 阻塞和非阻塞(Blocking and Non-blocking): 阻塞I/O是指当程序进行I/O操作时,如果没有数据可用或没有完成写入,程序将被阻塞,无法执行其他任务。非阻塞I/O允许程序在等待数据就绪时继续执行其他任务。

  5. 同步和异步(Synchronous and Asynchronous): 同步I/O是指程序在发起I/O操作后等待操作完成,然后继续执行后续任务。异步I/O允许程序发起I/O操作后继续执行其他任务,当操作完成时,程序会收到通知。

  6. 文件I/O: 文件I/O是指程序与文件系统进行交互的操作,包括读取文件内容、写入文件、创建文件、删除文件等。文件I/O通常用于数据的持久化和存储。

  7. 网络I/O: 网络I/O是指程序与其他计算机通过网络进行通信的操作,包括发送和接收数据、建立连接、断开连接等。网络I/O通常用于构建分布式应用程序。

I/O模型

不同的I/O模型使用不同的方法来处理输入和输出操作,具有不同的特性和适用场景。以下是常见的I/O模型:

  1. 阻塞I/O模型(Blocking I/O): 在阻塞I/O模型中,当应用程序发起一个I/O请求时,程序会被阻塞,直到操作完成或出错。这意味着程序需要等待,不能执行其他任务,直到I/O操作完成。阻塞I/O模型通常简单易用,但可能导致程序性能下降,特别是在高并发环境下。

  2. 非阻塞I/O模型(Non-blocking I/O): 非阻塞I/O模型允许应用程序在等待I/O操作完成时继续执行其他任务,而不会被阻塞。程序可以通过轮询或回调等方式来检查是否有数据可用或操作已完成。非阻塞I/O模型适用于需要高并发处理的场景,但编程复杂度较高。

  3. 多路复用I/O模型(Multiplexing I/O): 多路复用I/O模型使用了选择器(Selector)来监听多个I/O通道的状态,一旦某个通道准备好进行I/O操作,就会触发相应的事件,应用程序可以响应这些事件。这允许一个线程同时管理多个I/O通道,减少线程开销,提高并发性能。常见的多路复用I/O模型包括select、poll和epoll。

  4. 信号驱动I/O模型(Signal-driven I/O): 在信号驱动I/O模型中,应用程序通过注册信号处理程序来处理I/O事件。当I/O操作完成时,操作系统会向应用程序发送信号,应用程序可以捕获信号并处理事件。这种模型通常用于Unix系统。

  5. 异步I/O模型(Asynchronous I/O): 异步I/O模型允许应用程序发起I/O操作后继续执行其他任务,同时注册回调函数来处理I/O操作的完成。操作系统在I/O操作完成后调用回调函数通知应用程序。这种模型适用于需要高度并发和异步操作的场景。

BIO示例

BIO的简单流程

创建BIOServer

public class BIOServer {ServerSocket socketServer; // 服务端网络IO的封装对象// 构造服务器public BIOServer(int port) {try {socketServer = new ServerSocket(port);System.out.println("BIOServer start,Port :" + port);} catch (IOException e) {e.printStackTrace();}}
​// 端口有了,需要监听连接public void listen() {try {// 调用accept()阻塞等待客户端连接Socket client = socketServer.accept();System.out.println("communication port:" + client.getPort());
​InputStream inputStream = client.getInputStream(); // 客户端连接传递的信息流BufferedReader clientIn = new BufferedReader(new InputStreamReader(inputStream)); // 通过流读取客户端输入PrintWriter serverOut = new PrintWriter(client.getOutputStream()); // 基于socket构造服务端输出对象用于和client发送消息
​// 服务端通过控制台输入向客户端通信发送消息,模拟TCP的全双工通信BufferedReader systemIn = new BufferedReader(new InputStreamReader(System.in)); // 通过流读取控制台输入String line = systemIn.readLine();while (!line.equals("bye")) { // 持续连接serverOut.println(line); // 向客户端输出消息serverOut.flush(); // 手动将缓冲区中的数据强制刷新到输出流中,以确保数据被立即写入底层的输出流System.out.println("receive client msg:" + clientIn.readLine());System.out.println("server send msg:" + line);line = systemIn.readLine();}inputStream.close();clientIn.close();serverOut.close();client.close();socketServer.close();} catch (IOException e) {e.printStackTrace();}}
​public static void main(String[] args) throws IOException {// 启动服务端,开启监听new BIOServer(8080).listen();}
}

启动运行Server,创建Client连接Server

public class BIOClient {public static void main(String[] args){try {// 客户端网络IO的封装对象Socket socket = new Socket("localhost", 8080);
​// 控制台输入客户端输出信息BufferedReader systemIn = new BufferedReader(new InputStreamReader(System.in));// 在当前socket连接上写入数据PrintWriter clientOut = new PrintWriter(socket.getOutputStream(), true);// 获取服务端通过socket发送的输入流BufferedReader serverIn = new BufferedReader(new InputStreamReader(socket.getInputStream()));// 模拟Tcp全双工通信String line = systemIn.readLine();while (!line.equals("bye")) { // 持续连接clientOut.println(line); // 向服务端输出消息clientOut.flush(); // 手动将缓冲区中的数据强制刷新到输出流中,以确保数据被立即写入底层的输出流System.out.println("receive server msg:" + serverIn.readLine());System.out.println("client send msg:" + line);line = systemIn.readLine();}systemIn.close();serverIn.close();socket.close();} catch (IOException e) {e.printStackTrace();}}
}

BIO中的阻塞是指读写数据都是单向的,因此是同步阻塞。

BIO与NIO比较

流和通道最大的区别是流总是一个方向的,而通道是双向的,这也可以理解为阻塞和非阻塞的差异

BIONIO
阻塞非阻塞
面向流面向缓冲区
每个客户端连接都需要一个独立线程处理选择器,允许一个线程管理多个通道
连接少负载低,适合传统的同步阻塞式服务高并发性和响应性,适合聊天应用,实时游戏等
编程模型简单,高并发需要管理大量线程编程模型复杂,具备更好的性能和并发处理能力

NIO的三大核心

NIO有三大核心部分: Channel(通道),Buffer(缓冲区),Selector(选择器)

  1. 通道(Channel): 通道是NIO中的基本概念,代表着数据源和数据目标,可以是文件、网络连接、管道等。通道提供了一个可供程序读写数据的抽象接口,不同类型的通道可以用于不同的I/O操作。常见的通道类型包括:

    • FileChannel:用于文件I/O操作。

    • SocketChannel:用于套接字网络连接。

    • ServerSocketChannel:用于服务器套接字。

    • DatagramChannel:用于UDP协议数据报的通道。

  2. 缓冲区(Buffer): 缓冲区是NIO中用于存储数据的容器,通常是一个数组。缓冲区用于在通道和应用程序之间传输数据,可以读取数据到缓冲区或将数据从缓冲区写入通道。缓冲区提供了对数据的有序访问,常见的缓冲区类型包括:

    • ByteBuffer:用于存储字节数据。

    • CharBuffer:用于存储字符数据。

    • ShortBufferIntBufferLongBuffer:用于存储整数数据。

    • FloatBufferDoubleBuffer:用于存储浮点数数据。

  3. 选择器(Selector): 选择器是NIO中用于多路复用的关键组件,它允许一个线程管理多个通道。通过选择器,可以监视多个通道上的事件,如可读、可写等,并在事件就绪时唤醒相关的线程来处理。选择器的使用使得可以高效地处理多个并发连接,提高了程序的性能和响应性。

NIO核心之缓冲区

缓冲区(Buffer)一个用于特定基本数据类型的容器,所有缓冲区都是 Buffer 抽象类的子类。

Buffer就像一个数组,可以保存多个相同类型的数据。

Buffer常用子类:
  • ByteBuffer

  • CharBuffer

  • ShortBuffer

  • IntBuffer

  • LongBuffer

  • FloatBuffer

  • DoubleBuffer

创建一个容量为capacity 的 XxxBuffer 对象:

static XxxBuffer allocate(int capacity)
Buffer常用API
  • Buffer clear():清空缓冲区,重置位置和限制。

  • Buffer flip():切换缓冲区的读写模式,将位置设置为0,并将限制设置为当前位置。

  • Buffer rewind():将位置设置为0,保留限制的值,用于重读缓冲区中的数据。

  • Buffer mark():在当前位置设置标记。

  • Buffer reset():将位置重置为先前设置的标记位置。

  • boolean hasRemaining():检查是否还有剩余可读数据。

  • int remaining():返回剩余可读数据的数量。

  • int position():获取当前位置。

  • Buffer position(int n):将设置缓冲区的当前位置为 n.

  • int limit():获取限制值的位置。

  • Buffer limit(int newLimit):设置限制值的位置为n,并返回修改后的Buffer对象。

  • int capacity():返回 Buffer 的 capacity 大小

Buffer中的重要概念
  • 容量 (capacity) :作为一个内存块,Buffer具有一定的固定大小, 也称为"容量",缓冲区容量不能为负,并且创建后不能更改。

  • 限制 (limit):表示缓冲区中可以操作数据的大小 (limit 后数据不能进行读写)。缓冲区的限制不能 为负,并且不能大于其容量。 写入模式,限制等于 buffer的容量。读取模式下,limit等于写入的数据量。

  • 位置 (position):下一个要读取或写入的数据的索引。 缓冲区的位置不能为 负,并且不能大于其限制

位置、限制、容量遵守以下规律: 0 <= position <= limit <= capacity

1.init初始化容量为10的缓冲区

2.FileChannel.read(buffer)将一个大小为4byte的文件读入缓冲区(put操作)

3.flip()切换读写模式

4.buffer.get()读取缓冲区数据

5.clear()清空缓冲区,方便下次put操作

使用Buffer读写数据一般遵循以下四个步骤:

  1. 写入数据到Buffer

  2. 调用flip()方法,转换为读取模式

  3. 从Buffer中读取数据

  4. 调用buffer.clear()方法或者buffer.compact()方 法清除缓冲区

NIO核心之通道

Channel 类似于传统的“流”。只不过 Channel 本身不能直接访问数据,Channel 只能与 Buffer 进行交互。

FileChannel 类

获取通道的一种方式是对支持通道的对象调用getChannel() 方法。支持通道的类如下

  • FileInputStream

  • FileOutputStream

  • RandomAccessFile

  • DatagramSocket

  • Socket

  • ServerSocket

  • 获取通道的其他方式是使用 Files 类的静态方法 newByteChannel() 获取字节通道。或者通过通道的静态方法 open() 打开并返回指定通道

FileChannel常用方法
  • int read(ByteBuffer dst) :从Channel 到 中读取数据到 ByteBuffer

  • long read(ByteBuffer[] dsts) : 将Channel中的数据“分散”到 ByteBuffer[]

  • int write(ByteBuffer src) :将 ByteBuffer中的数据写入到 Channel

  • long write(ByteBuffer[] srcs) :将 ByteBuffer[] 到 中的数据“聚集”到 Channel

  • long position() :返回此通道的文件位置

  • FileChannel position(long p) :设置此通道的文件位置

  • long size() :返回此通道的文件的当前大小

  • FileChannel truncate(long s) :将此通道的文件截取为给定大小

  • void force(boolean metaData) :强制将所有对此通道的文件更新写入到存储设备中

NIO核心之选择器
概述

选择器(Selector)是SelectableChannle对象的多路复用器,Selector可以同时监控多个SelectableChannel的IO状况,也就是说,利用Selector可使一个单独的线程管理多个Channel。Selector是非阻塞IO的核心。

  • Java 的 NIO,用非阻塞的 IO 方式。可以用一个线程,处理多个的客户端连接,就会使用到 Selector(选择器)

  • Selector 能够检测多个注册的通道上是否有事件发生(注意:多个 Channel 以事件的方式可以注册到同一个(Selector),如果有事件发生,便获取事件然后针对每个事件进行相应的处理。这样就可以只用一个单线程去管

  • 理多个通道,也就是管理多个连接和请求。

  • 只有在连接/通道真正有读写事件发生时,才会进行读写,就大大地减少了系统开销,并且不必为每个连接都创建一个线程,不用去维护多个线程

  • 避免了多线程之间的上下文切换导致的开销

应用

创建 Selector :通过调用 Selector.open() 方法创建一个 Selector。

Selector selector = Selector.open();

向选择器注册通道:SelectableChannel.register(Selector sel, int ops)

//1. 获取通道
ServerSocketChannel ssChannel = ServerSocketChannel.open();
//2. 切换非阻塞模式
ssChannel.configureBlocking(false);
//3. 绑定连接
ssChannel.bind(new InetSocketAddress(port));
//4. 获取选择器
Selector selector = Selector.open();
//5. 将通道注册到选择器上, 并且指定“监听接收事件”
ssChannel.register(selector, SelectionKey.OP_ACCEPT);

当调用 register(Selector sel, int ops) 将通道注册选择器时,选择器对通道的监听事件,需要通过第二个参数 ops 指定。可以监听的事件类型(用 可使用 SelectionKey 的四个常量 表示):

  • 读 : SelectionKey.OP_READ (1)

  • 写 : SelectionKey.OP_WRITE (4)

  • 连接 : SelectionKey.OP_CONNECT (8)

  • 接收 : SelectionKey.OP_ACCEPT (16)

NIO非阻塞原理分析

Selector可以实现一个 I/O 线程可以并发处理 N 个客户端连接和读写操作,这从根本上解决了传统同步阻塞 I/O 一连接一线程模型,架构的性能、弹性伸缩能力和可靠性都得到了极大的提升。

服务端流程

1.三大核心配置

// 创建轮询器
Selector selector = Selector.open(); 
// 创建serverSocket通道
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open(); 
// 设置缓冲区
ByteBuffer buffer = ByteBuffer.allocate(1024); 

2.阻塞模式设置,绑定端口

// 绑定端口地址
serverSocketChannel.bind(new InetSocketAddress(8088)); 
// NIO为了兼容BIO,默认是阻塞式
serverSocketChannel.configureBlocking(false); 

3.注册通道并且指定“监听接收事件”

serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);

4.轮询的方式获取选择器上已经“准备就绪”的事件

public void listen() {try {while (selector.select() > 0) {Set<SelectionKey> selectedKeys = selector.selectedKeys(); // 获取已经准备好的事件Iterator<SelectionKey> keyIterator = selectedKeys.iterator(); // 迭代器轮询while (keyIterator.hasNext()) { // 同步体现在这里,因为每次只能拿一个key,每次只能处理一种状态SelectionKey key = keyIterator.next();process(key); // 处理key代表的业务:就绪、读、写等等keyIterator.remove();}}} catch (IOException e) {e.printStackTrace();}}
public void process(SelectionKey key) throws IOException {if(key.isAcceptable()) {// 接受客户端连接请求SocketChannel socketChannel = serverSocketChannel.accept();socketChannel.configureBlocking(false);
​// 将客户端socketChannel注册到selector,并监听读事件socketChannel.register(selector, SelectionKey.OP_READ);} else if (key.isReadable()) {// key.channel 从多路复用器中拿到客户端的引用SocketChannel socketChannel = (SocketChannel) key.channel();int len = socketChannel.read(buffer);if (len > 0) {buffer.flip(); // 读写切换,这里是切换为读模式String content = new String(buffer.array(), 0, len);System.out.println("get receive content:" + content);buffer.clear();}}}
客户端流程

1.获取通道

SocketChannel socketChannel = SocketChannel.open(new InetSocketAddress("localhost",8088));

2.切换为非阻塞模式

socketChannel.configureBlocking(false);

3.分配缓冲区大小

ByteBuffer buffer = ByteBuffer.allocate(1024);

4.发送数据给绑定的服务端

            Scanner scanner = new Scanner(System.in);while (true){System.out.print("请输入:");String msg = scanner.nextLine();buffer.put(msg.getBytes());buffer.flip();socketChannel.write(buffer);buffer.clear();}
简单NIO示例

服务端

public class Server {private int port;private Selector selector;private ServerSocketChannel serverSocketChannel;public Server(int port) {this.port = port;try {// 创建serverSocketChannel,初始化服务端口serverSocketChannel = ServerSocketChannel.open();serverSocketChannel.bind(new InetSocketAddress(this.port));// 创建selector选择器selector = Selector.open();// 设置非阻塞模式,NIO兼容BIO默认值为trueserverSocketChannel.configureBlocking(false);// 注册serverSocketChannel到选择器上,并监听客户端连接serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);System.out.println("server start on port:" + port);} catch (IOException e) {e.printStackTrace();}}public void listen(){try {while (selector.select() > 0) { // 有IO事件就绪,迭代处理// 获取SelectedKeys迭代器Iterator<SelectionKey> keyIterator = selector.selectedKeys().iterator();while (keyIterator.hasNext()) {SelectionKey currKey = keyIterator.next();// 根据IO事件状态判断处理process(currKey);keyIterator.remove();}}} catch (IOException e) {e.printStackTrace();}}/**deal with selectionKey读   : SelectionKey.OP_READ (1)写   : SelectionKey.OP_WRITE (4)连接 : SelectionKey.OP_CONNECT (8)接收 : SelectionKey.OP_ACCEPT (16)*/public void process(SelectionKey key) throws IOException {if (key.isValid()) {if (key.isAcceptable()) { // 处理新接入的请求System.out.println("done SelectionKey start:" + SelectionKey.OP_ACCEPT);ServerSocketChannel ssc = (ServerSocketChannel) key.channel();SocketChannel sc = ssc.accept();sc.configureBlocking(false);// 客户端SocketChannel连接到selectorsc.register(selector, SelectionKey.OP_READ);System.out.println("done SelectionKey end:" + SelectionKey.OP_ACCEPT);} else if (key.isReadable()) { // 读取数据System.out.println("done SelectionKey start:" + SelectionKey.OP_READ);SocketChannel sc = (SocketChannel) key.channel();ByteBuffer buffer = ByteBuffer.allocate(1024); // 分配缓冲区int len = sc.read(buffer);if (len > 0) {buffer.flip(); // 缓冲区读取切换String content = new String(buffer.array(), 0, len);System.out.println("server receive content :" + content);String currTime = content.equalsIgnoreCase("Query time") ? new Date(System.currentTimeMillis()).toString() : "error order";System.out.println("server send :" + currTime);doWrite(sc, currTime);} else if (len < 0) {// 对端链路关闭key.cancel();sc.close();} else {; // 读到0个字节,忽略}System.out.println("done SelectionKey end:" + SelectionKey.OP_READ);System.out.println("==============================================");}}}private void doWrite(SocketChannel sc, String currTime) throws IOException {if (currTime != null && currTime.trim().length() > 0) {byte[] bytes = currTime.getBytes();ByteBuffer buffer = ByteBuffer.allocate(bytes.length);buffer.put(bytes);buffer.flip();sc.write(buffer);}}public static void main(String[] args) {new Server(8088).listen();  // 启动服务端开启监听客户端连接}
}

客户端

public class Client {private int port;private SocketChannel socketChannel;private Selector selector;public Client(int port) {this.port = port;}public static void main(String[] args) {new Client(8088).run();}public void run() {try {// 创建socketChannelsocketChannel = SocketChannel.open(new InetSocketAddress("localhost", port));// 创建selectorselector = Selector.open();// 设置SocketChannel为非阻塞模式socketChannel.configureBlocking(false);//如果直连接连接成功,则注册到多路复用器上,并注册SelectionKey.OP_READ操作if (socketChannel.isConnected()) {socketChannel.register(selector, SelectionKey.OP_READ);} else {//如果直连接连接未成功,则注册到多路复用器上,并注册SelectionKey.OP_CONNECT操作socketChannel.register(selector, SelectionKey.OP_CONNECT);}while (socketChannel.isConnected()) {Scanner scanner = new Scanner(System.in);System.out.print("请输入:");String msg = scanner.nextLine();ByteBuffer buffer = ByteBuffer.allocate(msg.length());buffer.put(msg.getBytes());buffer.flip();socketChannel.write(buffer); // 客户端输入写入缓冲区System.out.println("client send :" + msg);//休眠1秒  无论是否有读写事件发生 selector每隔1秒被唤醒selector.select(1000);// 获取SelectedKeys迭代器Iterator<SelectionKey> keyIterator = selector.selectedKeys().iterator();while (keyIterator.hasNext()) {SelectionKey currKey = keyIterator.next();// 根据IO事件状态判断处理if (currKey.isValid()) {process(currKey);}keyIterator.remove();}}} catch (IOException e) {e.printStackTrace();}}/*** 管道状态* SelectionKey.OP_CONNECT 是否连接* SelectionKey.OP_ACCEPT  是否阻塞* SelectionKey.OP_READ    是否可读* SelectionKey.OP_WRITE  是否可写*/private static void process(SelectionKey key) throws IOException {SocketChannel sc = (SocketChannel) key.channel();if (key.isReadable()) {ByteBuffer bf = ByteBuffer.allocate(1024);int bytes = sc.read(bf);if (bytes > 0) {bf.flip();byte[] byteArray = new byte[bf.remaining()];bf.get(byteArray);String resopnseMessage = new String(byteArray, "UTF-8");System.out.println("接收到服务端数据:" + resopnseMessage);} else if (bytes < 0) {key.cancel();sc.close();}}}
}

运行结果:

AIO简介

AIO(Asynchronous I/O)是Java NIO(New I/O)库的一部分,用于实现异步非阻塞I/O操作。与传统的BIO(Blocking I/O)和NIO(Non-blocking I/O)不同,AIO通过回调机制在I/O操作完成时通知应用程序,而不需要应用程序一直轮询等待数据准备或操作完成。

以下是Java AIO的简介和主要特点:

主要特点
  1. 异步操作: AIO提供了异步I/O操作,允许应用程序在发起I/O请求后继续执行其他任务,而不必等待I/O操作完成。

  2. 回调机制: AIO使用回调机制通知应用程序有关I/O操作的完成情况。应用程序可以注册回调函数,当I/O操作完成时,操作系统会调用回调函数来处理数据。

  3. 适用于高并发: AIO适用于高并发环境,可以处理大量的并发连接和I/O操作,而无需为每个连接创建一个线程。

  4. 文件I/O和网络I/O: AIO不仅适用于网络编程,还适用于文件I/O。它可以用于处理文件读写、网络套接字通信等多种I/O操作。

主要组件和类
  1. AsynchronousChannel: AIO的通道类,包括AsynchronousFileChannel(文件I/O)和AsynchronousSocketChannel(套接字网络通信)等。这些通道支持异步I/O操作。

  2. CompletionHandler: 用于注册回调函数的接口,包括CompletedHandler<V, A>,其中V表示I/O操作的结果类型,A表示附加的上下文对象。应用程序通过实现这个接口来处理I/O操作完成的通知。

  3. AsynchronousServerSocketChannel: 用于实现异步的服务器套接字通信,可以监听并接受连接请求。

总结

BIO(Blocking I/O)、NIO(Non-blocking I/O)和AIO(Asynchronous I/O)各自适用于不同的场景,以下是它们的主要适用场景:

  1. BIO(Blocking I/O):

    • 适用场景: BIO适用于连接数量较少且每个连接的负载较低的场景。它在每个连接上都创建一个独立的线程来处理I/O操作,因此不适合高并发的环境。

    • 示例应用: 传统的同步阻塞式服务器,如Web服务器的早期版本,FTP服务器等。

  2. NIO(Non-blocking I/O):

    • 适用场景: NIO适用于高并发和高吞吐量的网络应用,其中需要处理大量并发连接。它通过复用少量线程来管理多个通道,减少了线程开销,提高了性能和资源利用率。

    • 示例应用: 高性能的网络服务器、聊天应用、实时游戏服务器、代理服务器等。

  3. AIO(Asynchronous I/O):

    • 适用场景: AIO适用于需要高并发、异步处理的应用,尤其是在处理文件I/O或需要等待时间较长的I/O操作时。它通过回调机制在I/O操作完成时通知应用程序,允许应用程序继续执行其他任务,不必等待I/O操作完成。

    • 示例应用: 文件上传/下载服务、邮件服务器、高吞吐量的文件传输应用等。

综合考虑,选择适当的I/O模型取决于应用程序的需求和性能要求。以下是一些指导原则:

  • 如果应用程序的并发连接数量较低,且可以接受阻塞模式,BIO可能是一个简单的选择。

  • 如果应用程序需要高并发处理,能够轻松处理数千个并发连接,NIO是更好的选择。

  • 如果应用程序需要异步处理且需要高并发,AIO可能是更好的选择,尤其在处理文件I/O或等待时间较长的I/O操作时。

另外,不同的应用程序可能会使用混合模型,根据具体需求在不同的部分使用BIO、NIO或AIO,以充分发挥它们的优势。

相关文章:

聊聊分布式架构05——[NIO基础]BIO到NIO的演进

目录 I/O I/O模型 BIO示例 BIO与NIO比较 NIO的三大核心 NIO核心之缓冲区 Buffer常用子类&#xff1a; Buffer常用API Buffer中的重要概念 NIO核心之通道 FileChannel 类 FileChannel常用方法 NIO核心之选择器 概述 应用 NIO非阻塞原理分析 服务端流程 客户端…...

聊天、会议、多媒体一体化:多平台支持的即时通讯系统 | 开源日报 No.44

harness/gitness Stars: 28.2k License: Apache-2.0 Gitness 是一个建立在 Drone 之上的新型开源开发者平台&#xff0c;具备代码托管和流水线功能。它提供了以下核心优势&#xff1a; 轻量级、超快速的代码托管和持续集成服务支持 Docker 容器化部署可以在本地环境中构建和…...

收录一些常见的算法题型

常用算法 字符串 s.trim():去掉字符串首尾的空格s.split("\\s"):按照空格对字符串分割 树 前中后序遍历 /*** 统一一下* param root* return*///前序public static List<Integer> preOrder(TreeNode root){List<Integer> list new ArrayList();Stac…...

Node-RED系列教程-25node-red获取天气

安装节点:node-red-contrib-weather 节点图标如下: 使用说明:node-red-contrib-weather (node) - Node-RED 流程图中填写经度和纬度即可。 演示: json内容: {...

Rust中的枚举和模式匹配

专栏简介&#xff1a;本专栏作为Rust语言的入门级的文章&#xff0c;目的是为了分享关于Rust语言的编程技巧和知识。对于Rust语言&#xff0c;虽然历史没有C、和python历史悠远&#xff0c;但是它的优点可以说是非常的多&#xff0c;既继承了C运行速度&#xff0c;还拥有了Java…...

好物周刊#19:开源指北

https://github.com/cunyu1943/JavaPark https://yuque.com/cunyu1943 村雨遥的好物周刊&#xff0c;记录每周看到的有价值的信息&#xff0c;主要针对计算机领域&#xff0c;每周五发布。 一、项目 1. Vditor 一款浏览器端的 Markdown 编辑器&#xff0c;支持所见即所得、…...

分布式数据库(林子雨慕课课程)

文章目录 4. 分布式数据库HBase4.1 HBase简介4.2 HBase数据模型4.3 HBase的实现原理4.4 HBase运行机制4.5 HBase的应用方案4.6 HBase安装和编程实战 4. 分布式数据库HBase 4.1 HBase简介 HBase是BigTable的开源实现 对于网页搜索主要分为两个阶段 1.建立整个网页索引&#xf…...

使用UiPath和AA构建的解决方案 3. CRM 自动化

您是否曾经从一个应用程序中查找数据并更新另一个系统? 在许多情况下,人们在系统之间复制和移动数据。有时,可能会发生“转椅活动”,从而导致人为失误。RPA可以帮助我们自动化这些活动,使其更快,同时还可以消除任何人为错误。 在这个项目中,我们将在客户服务中自动化一…...

【C++设计模式之状态模式:行为型】分析及示例

简介 状态模式&#xff08;State Pattern&#xff09;是一种行为型设计模式&#xff0c;它允许对象在内部状态改变时改变其行为&#xff0c;看起来就像是改变了其类。状态模式将对象的状态封装成不同的类&#xff0c;并使得对象在不同状态下有不同的行为。 描述 状态模式通过…...

微信小程序使用路由传参和传对象的方法

近期在做微信小程序开发&#xff0c;在页面跳转时&#xff0c;需要携带参数到下一个页面&#xff0c;尤其是将对象传入页面。为了方便重温&#xff0c;特此记录。 路由传字符串参数 原始页面 传递字符串参数比较简单。路由跳转有两种方式&#xff0c;一种是通过navigator组件…...

中国创可贴市场研究与未来预测报告(2023版)

内容简介&#xff1a; 创可贴由胶布&#xff08;带&#xff09;、吸水垫、防粘层等组成&#xff0c;胶布以弹性布、棉布、无纺布或PE、PVC、PU打孔膜、TPU等材料为常见基材&#xff0c;涂以氧化锌和橡胶为主要原料的胶浆或医用压敏胶黏剂或丙烯酸酯胶粘剂制成。 目前中国主要…...

水库安全监测方案(实时数据采集、高速数据传输)

​ 一、引言 水库的安全监测对于防止水灾和保障人民生命财产安全至关重要。为了提高水库安全监测的效率和准确性&#xff0c;本文将介绍一种使用星创易联DTU200和SG800 5g工业路由器部署的水库安全监测方案。 二、方案概述 本方案主要通过使用星创易联DTU200和SG800 5g工业路…...

vue项目 ueditor使用示例

简介 UEditor是由百度Web前端研发部开发的所见即所得富文本web编辑器&#xff0c;具有轻量&#xff0c;功能丰富&#xff0c;易扩展等特点。UEditor支持常见的文本编辑功能&#xff0c;如字体、颜色、大小、加粗、斜体、下划线、删除线等&#xff0c;同时还支持超链接、图片上…...

深度学习笔记之优化算法(四)Nesterov动量方法的简单认识

机器学习笔记之优化算法——Nesterov动量方法的简单认识 引言回顾&#xff1a;梯度下降法与动量法Nesterov动量法Nesterov动量法的算法过程描述总结 引言 上一节对动量法进行了简单认识&#xff0c;本节将介绍 Nesterov \text{Nesterov} Nesterov动量方法。 回顾&#xff1a;…...

比 N 小的最大质数

系列文章目录 进阶的卡莎C++_睡觉觉觉得的博客-CSDN博客数1的个数_睡觉觉觉得的博客-CSDN博客双精度浮点数的输入输出_睡觉觉觉得的博客-CSDN博客足球联赛积分_睡觉觉觉得的博客-CSDN博客大减价(一级)_睡觉觉觉得的博客-CSDN博客小写字母的判断_睡觉觉觉得的博客-CSDN博客纸币(…...

JavaScript 生成随机颜色

代码 function color(color) {return (color "0123456789abcdef"[Math.floor(Math.random() * 6)]) && (color.length 6 ? color : arguments.callee(color)); }使用 // 用法1&#xff1a;全部随机生成 "#" color(""); // #201050…...

Savepoints

语法 SAVEPOINT 名称 RELEASE SAVEPOINT 名称 ROLLBACK TRANSACTION TO SAVEPOINT 名称 Savepoints 与BEGIN和COMMIT类似的创建事务的方法&#xff0c;名称不要求唯一且可以嵌套使用。 可以用在BEGIN…COMMIT定义的事务内部或外部。当在外部时&#xff0c;最外层的savepoin…...

【MySQL】基本查询(二)

文章目录 一. 结果排序二. 筛选分页结果三. Update四. Delete五. 截断表六. 插入查询结果结束语 操作如下表 //创建表结构 mysql> create table exam_result(-> id int unsigned primary key auto_increment,-> name varchar(20) not null comment 同学姓名,-> chi…...

Qt:多语言支持,构建全面应用程序“

Qt&#xff1a;强大API、简化框架、多语言支持&#xff0c;构建全面应用程序" 强大的API&#xff1a;Qt提供了丰富的API&#xff0c;包括250多个C类&#xff0c;基于模板的集合、序列化、文件操作、IO设备、目录管理、日期/时间等功能。还包括正则表达式处理和支持2D/3D…...

性能监控-链路级监控工具

常见的链路监控工具&#xff0c;我们都称之为 APM 开源工具 几个开源的好用的工具&#xff0c;它们分别是 Pinpoint、SkyWalking、Zipkin、CAT 网络上也有人对这几个工具做过测试 比对&#xff0c;得到的结论是每个产品对性能的影响都在 10% 以下&#xff0c;其中 SkyWalking …...

clickonce 程序发布到ftp在使用cnd 加速https 支持下载,会不会报错

ClickOnce 是一种用于发布和部署.NET应用程序的技术&#xff0c;通常用于本地部署或通过网络分发应用程序。将 ClickOnce 程序发布到 FTP 服务器并使用 CDN&#xff08;内容分发网络&#xff09;进行加速是可能的&#xff0c;但要确保配置正确以避免出现错误。 在使用 CDN 加速…...

Nginx详细学习记录

1. Nginx概述 Nginx是一个轻量级的高性能HTTP反向代理服务器&#xff0c;同时它也是一个通用类型的代理服务器&#xff0c;支持绝大部分协议&#xff0c;如TCP、UDP、SMTP、HTTPS等。 1.1 Nginx基础架构 Nginx默认采用多进程工作方式&#xff0c;Nginx启动后&#xff0c;会运行…...

golang gin——中间件编程以及jwt认证和跨域配置中间件案例

中间件编程jwt认证 在不改变原有方法的基础上&#xff0c;添加自己的业务逻辑。相当于grpc中的拦截器一样&#xff0c;在不改变grpc请求的同时&#xff0c;插入自己的业务。 简单例子 func Sum(a, b int) int {return a b }func LoggerMiddleware(in func(a, b int) int) f…...

如何快速制作令人惊叹的长图海报

在当今的数字时代&#xff0c;制作一张吸引人的长图海报已成为许多人的需求。无论是为了宣传活动&#xff0c;还是展示产品&#xff0c;一张设计精美的长图海报都能引起人们的注意。下面&#xff0c;我们将介绍一种简单的方法&#xff0c;使用在线海报制作工具来创建长图海报。…...

D (1092) : DS循环链表—约瑟夫环(Ver. I - A)

Description N个人坐成一个圆环&#xff08;编号为1 - N&#xff09;&#xff0c;从第S个人开始报数&#xff0c;数到K的人出列&#xff0c;后面的人重新从1开始报数。依次输出出列人的编号。 例如&#xff1a;N 3&#xff0c;K 2&#xff0c;S 1。 2号先出列&#xff0c;然…...

Python爬虫(二十二)_selenium案例:模拟登陆豆瓣

本篇博客主要用于介绍如何使用seleniumphantomJS模拟登陆豆瓣&#xff0c;没有考虑验证码的问题&#xff0c;更多内容&#xff0c;请参考&#xff1a;Python学习指南 #-*- coding:utf-8 -*-from selenium import webdriver from selenium.webdriver.common.keys import Keysimp…...

1. Flink程序打Jar包

文章目录 步骤注意事项 步骤 用 maven 打 jar 包&#xff0c;需要在 pom.xml 文件中添加打包插件依赖 <build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-shade-plugin</artifactId><ver…...

水波纹文字效果动画

效果展示 CSS 知识点 text-shadow 属性绘制立体文字clip-path 属性来绘制水波纹 工具网站 CSS clip-path maker 效果编辑器 页面整体结构实现 使用多个 H2 标签来实现水波纹的效果实现&#xff0c;然后使用clip-path结合动画属性一起来进行波浪的起伏动画实现。 <div …...

【1.1】神经网络:关于神经网络的介绍

✅作者简介&#xff1a;大家好&#xff0c;我是 Meteors., 向往着更加简洁高效的代码写法与编程方式&#xff0c;持续分享Java技术内容。 &#x1f34e;个人主页&#xff1a;Meteors.的博客 &#x1f49e;当前专栏&#xff1a; 神经网络&#xff08;随缘更新&#xff09; ✨特色…...

java项目中git的.ignore文件设置

在Git中&#xff0c;ignore是用来指定Git应该忽略的故意不被追踪的文件。它并不影响已经被Git追踪的文件。我们可以通过.ignore文件在Git中指定要忽略的文件。 当我们执行git add命令时&#xff0c;Git会检查.gitignore文件&#xff0c;并自动忽略这些文件和目录。这样可以避免…...

嘉兴高端网站建设/百度推广登录平台

在上一文中&#xff0c;论述两个.Net Framework对null应用不够合理的例子。大家评论中&#xff0c;给出了不少指导性意见&#xff0c;这里也对.Net中null的使用规范作一下总结。 1. Empty代表瓶子是空的&#xff0c;null代表瓶子都没有 首先要明确你的“瓶子”是什么&#xff0…...

房山网站建设服务/seo优化常识

说明&#xff1a; 本文原创作者『Allen5G』 首发于微信公众号『Allen5G』&#xff0c;同时也更新在我的&#xff1a;CSDN&#xff0c;简书 标签&#xff1a;嵌入式软件&#xff0c;算法&#xff0c;通信 上拉电路 --- 应用于总线或者开漏模式&#xff0c;提供稳定电压 码字不…...

web前端期末考试网页制作/快速提升排名seo

可编辑版Word完美格式HUNAN CITY UNIVERSITY数据库系统课程设计设计题目&#xff1a;宿舍管理信息系统姓 名&#xff1a;学 号&#xff1a;专 业&#xff1a;信息与计算科学指导教师&#xff1a;20年 12月1日目 录TOC \o "1-3" \h \z HYPERLINK \l "_Toc2306871…...

内江市住房和城乡建设局网站电话号码/一份完整的电商运营方案

重要提示&#xff1a; 下一版本的 Microsoft SQL Server 将删除该功能。请不要在新的开发工作中使用该功能&#xff0c;并尽快修改当前还在使用该功能的应用程序。 改为使用链接服务器和链接服务器存储过程。 返回远程 SQL Server 数据库服务器在登录记录中显示的名称。 Transa…...

做翻译 网站/百度网站网址是多少

KD302 成本中心 CTR xxx/xxxx, 成本要素 4210000: 不能划分 (2013-03-05 11:11:17) 转载▼ 标签&#xff1a; it 分类&#xff1a; SAP 都什么年代了&#xff0c;版本都ehp6了还有这个BUG啊。。。 成本中心 CTR xxx/xxxx, 成本要素 4210000: 不能划分 消息号 KD302 诊断 …...

做调查的网站推荐/石家庄关键词优化平台

对于一个网站刚上线&#xff0c;没人知道怎么办?登陆BAIDU、GOOGLE等各大搜索引擎提交你的网站信息吗?友情链接比你到各大搜索引擎提交来的更快。友情链接&#xff1a;网站之间链接互换是互相推广的一种重要方式。这样可以加深网站之间资源共享、用户共享、互相推荐等&#x…...