Netty入门基础:IO模型中BIO\NIO概念及区别【附演示代码】
文章目录
- 😀BIO
- 💢实战demo
- 🌈NIO
- 🏍Buffer
- 核心属性
- 核心方法
- 🎗Channel
- 🎈Selector
- 核心方法
- 🧨实战demo
- 🎨粘包与半包
😀BIO
传统IO模型,同步阻塞,每个来自客户端的连接,服务端就专门启动一个线程进行处理,如果这个连接不做任何事情,会造成不必要的线程开销。
适用于连接数目小且固定的架构,这种方式对服务器资源要求比较高,并发局限于应用中,JDK1.4 以前的唯一选择,程序简单易理解。
💢实战demo
验证在BIO模型下,服务端中一个线程只能处理一个客户端的连接。
服务端代码,使用SocketChannel,监听9090端口。
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class BIOServer {public static void main(String[] args) throws IOException {// 服务端监听端口try (ServerSocket serverSocket = new ServerSocket(9090)) {System.out.println("<<服务端>> 等待连接中...");while (true) {// 监听与此 Socket 建立的连接并接受它。该方法阻塞,直到建立连接。Socket socket = serverSocket.accept();System.out.printf("<<服务端>> 收到来自%s的连接\n", socket.getRemoteSocketAddress());handler(socket);}}}//编写一个handler方法,和客户端通讯public static void handler(Socket socket) throws IOException {byte[] bytes = new byte[1024];// 通过socket获取输入流InputStream inputStream = socket.getInputStream();// 循环的读取客户端发送的数据while (true) {int read = inputStream.read(bytes);if (read != -1) {String msg = new String(bytes, 0, read);System.out.printf("当前线程id = %s,线程名字=%s。", Thread.currentThread().getId(), Thread.currentThread().getName());System.out.printf("接受到来自%s的消息:", socket.getRemoteSocketAddress());System.out.println(msg);} else {System.out.printf("关闭和%s的连接\n", socket.getRemoteSocketAddress());break;}}}
}
客户端代码,使用Socket连接服务端。
import java.io.*;
import java.net.Socket;
import java.util.Scanner;public class BIOClient {public static void main(String[] args) throws IOException {Socket socket = new Socket("localhost", 9090);System.out.printf("当前 <<客户端>> 地址为%s\n", socket.getLocalSocketAddress());System.out.print("请输入内容,发送到服务端 (输出“exit”时退出):");Scanner scanner = new Scanner(System.in);while(scanner.hasNext()) {String msg = scanner.nextLine();if ("exit".equalsIgnoreCase(msg)) {socket.close();break;}OutputStream outputStream = socket.getOutputStream();outputStream.write(msg.getBytes());System.out.print("请输入内容,发送到服务端 (输出“exit”时退出):");}}
}
测试:先启动服务端,再启动两个客户端,分别发送消息。
发现只有第一个客服端连接到服务端,其实这时第二个客户端已经建立连接,但是因为BIO模型,服务端只能处理一个连接,当关闭第一个客户端后,第二个客户端的消息就会马上发送到服务端了。


🌈NIO
NIO的三大核心组件关系图

一个线程关联一个Selector。
一个Selector关联多个Channel,Selector根据不同的事件在各个Channel中切换。
Channel通过Buffer在服务端和客户端之间进行数据交换。
🏍Buffer
Buffer对象本质上是一个可读写数据的内存块,一个容器(数组)。
Buffer不同于BIO中流,BIO中同一个流只能进行写或者读的操作。但是Buffer既可以写入数据也可以读取数据
Buffer种类有以下几种,其中使用较多的是ByteBuffer

核心属性
// Invariants: mark <= position <= limit <= capacity
private int mark = -1;
private int position = 0;
private int limit;
private int capacity;
- capacity:缓冲区的容量。通过构造函数赋予,一旦设置,无法更改
- limit:缓冲区的界限。位于limit 后的数据不可读写。缓冲区的限制不能为负,并且不能大于其容量
- position:下一个读写位置的索引(类似PC)。缓冲区的位置不能为负,并且不能大于limit
- mark:记录当前position的值。position被改变后,可以通过调用reset() 方法恢复到mark的位置。
核心方法
put(T obj):插入数据,同时position往后移动。
flip():读写模式的切换。本质是改变核心属性的值。
get():读取缓存区中的一个值,同时position往后移动。
get(int index):读取指定位置的值,但是position不会变。
rewind():只在读模式下使用,恢复position、limit和capacity的值,变为进行get()前的值
clean():将缓冲区的属性恢复最初的状态,达到删除的效果,此时原数据还在,下次写会覆盖。
mark():将position的值保存到mark属性。
reset():将mark属性的值给position
compact():ByteBuffer类的方法。把position之前的数据清空,把剩余的数据往前移动。
🎗Channel
- Channel不同于BIO中流,Channel可以读写,但流只能读或只能写。
- Channel与Buffer紧密结合,数据总是从Channel读入Buffer,从Buffer写入Channel。
- 常见的通道类型包括FileChannel(用于文件I/O)、SocketChannel(用于网络I/O)和ServerSocketChannel(用于服务器端网络I/O)。
🎈Selector
Selector能够检测多个注册的Channel上是否有事件发生,如果有事件发生,便获取事件然后针对每个事件进行相应的处理。这样就可以只用一个单线程去管理多个Channel,也就是管理多个连接和请求。
核心方法
public abstract class Selector implements Closeable {// 创建一个Selector并返回public static Selector open() throws IOException;// Selector是否打开public abstract boolean isOpen();// 返回创建Selector的提供者public abstract SelectorProvider provider();// 返回的Selector的 key 集合public abstract Set<SelectionKey> keys();// 返回Selector选择过的key集合public abstract Set<SelectionKey> selectedKeys();// Selector立即执行选择操作,返回已选择的key的数量// 选择操作:对注册进入Selector中的Channel(准备进行IO操作)// 放到内部的一个集合中。public abstract int selectNow() throws IOException;// Selector阻塞执行选择操作,直到有可以选择的Channel// 或者阻塞时间超过timeout,才会返回。public abstract int select(long timeout) throws IOException;// Selector阻塞执行选择操作,直到有可以选择的Channel才会返回。public abstract int select() throws IOException;// 使尚未返回的第一个选择操作立即返回。public abstract Selector wakeup();// 关闭Selectorpublic abstract void close() throws IOException;
}
🧨实战demo
验证在NIO模型下,服务端一个线程能处理多个客户端连接。
public class NIOServer {public static void main(String[] args) throws Exception {try(Selector selector = Selector.open();ServerSocketChannel serverSocketChannel = ServerSocketChannel.open()) {serverSocketChannel.bind(new InetSocketAddress(9090));serverSocketChannel.configureBlocking(false);serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);System.out.println("<<服务端>> 等待连接中...");while(true) {// (阻塞)选择已准备好进行IO操作的Channel对应的key集合int count = selector.select();if(count > 0) {// 返回前面选择的key集合,select()必须在selectedKeys()之前调用,否则没有选择key,那么selectedKeys()就没有数据Iterator<SelectionKey> iterator = selector.selectedKeys().iterator();while(iterator.hasNext()) {SelectionKey selectionKey = iterator.next();iterator.remove();// 测试key对应的Channel是否准备好接受一个新的Socket连接if(selectionKey.isAcceptable()) {// 拿到Socket连接SocketChannel socketChannel = serverSocketChannel.accept();socketChannel.configureBlocking(false);// 注册进入SelectorsocketChannel.register(selector, SelectionKey.OP_READ);System.out.printf("<<服务端>> 收到来自%s的连接\n", socketChannel.getRemoteAddress());} else if (selectionKey.isReadable()) { // 测试key的通道是否已准备好读取。readData(selectionKey);}}}}}}private static void readData(SelectionKey selectionKey) throws IOException {//拿到key关联的SocketChannelSocketChannel socketChannel = (SocketChannel) selectionKey.channel();ByteBuffer byteBuffer = ByteBuffer.allocate(1024);// 把Channel中数据写入Bufferint count = socketChannel.read(byteBuffer);if(count > 0) {// 反转Buffer,切换Buffer的读写模式byteBuffer.flip();String msg = new String(byteBuffer.array(), 0, byteBuffer.limit());System.out.printf("当前线程id = %s,线程名字=%s。", Thread.currentThread().getId(), Thread.currentThread().getName());System.out.printf("接受到来自%s的消息:", socketChannel.getRemoteAddress());System.out.println(msg);} else if (count == -1) {System.out.printf("关闭和%s的连接\n", socketChannel.getRemoteAddress());selectionKey.cancel();socketChannel.close();}}
}
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;
import java.util.Scanner;public class NIOClient {public static void main(String[] args) throws IOException {SocketChannel socketChannel = SocketChannel.open();socketChannel.connect(new InetSocketAddress("localhost", 9090));Scanner scanner = new Scanner(System.in);System.out.printf("当前 <<客户端>> 地址为%s\n", socketChannel.getLocalAddress());System.out.print("请输入内容,发送到服务端 (输出“exit”时退出):");while (scanner.hasNext()) {String msg = scanner.nextLine();if ("exit".equalsIgnoreCase(msg)) {socketChannel.close();break;}socketChannel.write(ByteBuffer.wrap(msg.getBytes()));System.out.print("请输入内容,发送到服务端 (输出“exit”时退出):");}}
}
测试:先启动服务端,再启动两个客户端,分别发送消息。
发现两个客户端可以同时连接到服务端,同时发送消息。


🎨粘包与半包
粘包:发送方在发送数据时,并不是一条一条地发送数据,而是将数据整合在一起,当数据达到一定的数量后再一起发送。这就会导致多条信息被放在一个缓冲区中被一起发送出去
半包:接收方的缓冲区的大小是有限的,当接收方的缓冲区满了以后,就需要将信息截断,等缓冲区空了以后再继续放入数据。这就会发生一段完整的数据最后被截断的现象
相关文章:
Netty入门基础:IO模型中BIO\NIO概念及区别【附演示代码】
文章目录 😀BIO💢实战demo 🌈NIO🏍Buffer核心属性核心方法 🎗Channel🎈Selector核心方法 🧨实战demo 🎨粘包与半包 😀BIO 传统IO模型,同步阻塞,每…...
vue2 使用环境变量
一. 在根目录下创建.env.xxx文件 .env 基础系统变量,无论何种环境,都可使用其中配置的值,其他环境中的变量会覆盖.env中的同名变量。 .env.development 开发环境 .env.production 生产环境 .env.staging 测试环境 二. 内容格式 vue2 使用是以…...
数据预处理
继续提取代码片段: 12. **导入iris数据集并查看前5行数据**: python from sklearn.datasets import load_iris iris load_iris() X iris.data print(iris数据集的维度为:, X.shape) print(iris数据集的前5行数据为:\n, X[:5]) …...
django宠物领养管理系统-计算机毕业设计源码26858
目录 1 绪论 1.1 选题背景与意义 1.2国内外研究现状 1.3论文结构与章节安排 2系统分析 2.1 可行性分析 2.2 系统流程分析 2.2.1 数据流程 3.3.2 业务流程 2.3 系统功能分析 2.3.1 功能性分析 2.3.2 非功能性分析 2.4 系统用例分析 2.5本章小结 3 系统总体设计 3…...
使用TeamViewer远程局域网内的两台电脑
有个场景,有人还不知道TV可以局域网操作,记录一下。 主要就是修改设置,将取消激活改为接受 然后输入受控端的ip即可...
GUI简介、Swing的常用组件、java程序的运行过程、class文件、JAR、runable_jar、双括号初始化
GUI简介 GUI:图形用户界面,在计算机中采用图形的方式显示用户界面 java的GUI开发 AWT:java最早推出的GUI编程开发包,界面风格跟随操作系统SWT:eclipse就是java使用SWT开发的Swing:在AWT的基础上扩充了功能…...
@Autowired和@Resource和getBean()区别
今天遇到一个对我来说很奇葩的错误,我想在Service中注入bean,我这里使用了Autowired和Resource都不能注入,导致初始化失败,使用了getBean()方法就可以注入。从来没有遇到过这个问题。后来我查询了一下,才明白了原理。我…...
Merlion笔记(四):添加一个新的预测模型
文章目录 1 模型配置类2 模型类3 运行模型:一个简单的例子4 可视化5 定量评估6 定义一个基于预测器的异常检测器 本文提供了一个示例,展示如何向 Merlion 添加一个新的预测模型,遵循 CONTRIBUTING.md 中的说明。建议在阅读本篇文章之前,先查…...
【论文阅读】ESRGAN
学习资料 论文题目:增强型超分辨率生成对抗网络(ESRGAN: Enhanced Super-Resolution Generative Adversarial Networks)论文地址:[1809.00219] ESRGAN:增强型超分辨率生成对抗网络代码:xinntao / ESRGAN&am…...
电脑异常情况总结
文章目录 笔记本无症状息屏黑屏 笔记本无症状息屏黑屏 🍎 问题描述: 息屏导致黑屏;依次操作计算机--》右键--》管理--》事件查看器--》Windows日志--》系统;从息屏到异常黑屏之间出现了很多错误,如下:事件…...
[项目详解][boost搜索引擎#1] 概述 | 去标签 | 数据清洗 | scp
目录 一、前言 二、项目的相关背景 三、搜索引擎的宏观原理 四、搜索引擎技术栈和项目环境 五、正排索引 VS 倒排索引--原理 正排索引 分词 倒排索引 六、编写数据去除标签和数据清洗模块 Parser 1.数据准备 parser 编码 1.枚举文件 EnumFile 2.去标签ParseHtml(…...
PL/I语言的起源?有C语言,有B语言和A语言吗?为什么shell脚本最开始可能有#!/bin/bash字样?为什么不支持嵌套注释?
PL/I语言的起源 在20世纪50~60年代,当时主流的编程语言是COBOL/FORTRAN/ALGOL等,IBM想要设计一门通用的编程语言,已有的编程语言无法实现此要求,故想要设计一门新语言,即是PL/I. PL/I是Programming Language/One的缩写…...
gin入门教程(3):创建第一个 HTTP 服务器
首先设置golang github代理,可解决拉取git包的时候,无法拉取的问题: export GOPROXYhttps://goproxy.io再查看自己的go版本: go version我这里的版本是:go1.23.2 linux/arm64 准备工作做好之后就可以进行开发了 3.…...
Vue+ECharts+iView实现大数据可视化大屏模板
Vue数据可视化 三个大屏模板 样式还是比较全的 包括世界地图、中国地图、canvas转盘等 项目演示: 视频: vue大数据可视化大屏模板...
el-table 表格设置必填项
el-table 表格设置必填项 要在 el-table 中集成 el-form 来设置必填项,并进行表单验证,可以使用 Element UI 提供的表单验证功能。下面是一个详细的示例,展示了如何在 el-table 中使用 el-form 来设置必填项,并进行验证。 示例代…...
vivo 轩辕文件系统:AI 计算平台存储性能优化实践
在早期阶段,vivo AI 计算平台使用 GlusterFS 作为底层存储基座。随着数据规模的扩大和多种业务场景的接入,开始出现性能、维护等问题。为此,vivo 转而采用了自研的轩辕文件系统,该系统是基于 JuiceFS 开源版本开发的一款分布式文件…...
Vue学习笔记(四)
事件处理 我们可以使用 v-on 指令 (通常缩写为 符号) 来监听 DOM 事件,并在触发事件时执行一些 JavaScript。用法为 v-on:click"methodName" 或使用快捷方式 click"methodName" 事件处理器的值可以是: 内联事件处理器࿱…...
发送短信,验证码
短信 注册阿里云的账号 开通短信服务 测试短信服务是否可用 导入jar <!-- 短信相关 --><dependency><groupId>com.aliyun</groupId><artifactId>aliyun-java-sdk-core</artifactId><version>4.6.0</version><…...
国内大语言模型哪家更好用?
大家好,我是袁庭新。 过去一年,AI大语言模型在爆发式增长,呈现百家争鸣之态。国内外相关厂商积极布局,并相继推出自家研发的智能化产品。 我在工作中已习惯借助AI来辅助完成些编码、创作、文生图等任务,甚至对它们产…...
OTP一次性密码、多因子认证笔记
文章目录 双因子认证(多因子认证)otp算法(ONE-TIME PASSWORD)otp算法大概分为几部 otp的机制服务端客户端(app端)两种主流算法otp流程图 otp是通用的吗 手机验证码天天在用,但是居然不知道这个是otp,伤自尊了,必须弄清原理。 先要知道几个概念…...
Python爬虫实战:研究MechanicalSoup库相关技术
一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...
接口测试中缓存处理策略
在接口测试中,缓存处理策略是一个关键环节,直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性,避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明: 一、缓存处理的核…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...
相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了: 这一篇我们开始讲: 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下: 一、场景操作步骤 操作步…...
使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...
Objective-C常用命名规范总结
【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名(Class Name)2.协议名(Protocol Name)3.方法名(Method Name)4.属性名(Property Name)5.局部变量/实例变量(Local / Instance Variables&…...
多模态商品数据接口:融合图像、语音与文字的下一代商品详情体验
一、多模态商品数据接口的技术架构 (一)多模态数据融合引擎 跨模态语义对齐 通过Transformer架构实现图像、语音、文字的语义关联。例如,当用户上传一张“蓝色连衣裙”的图片时,接口可自动提取图像中的颜色(RGB值&…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...
Ascend NPU上适配Step-Audio模型
1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统,支持多语言对话(如 中文,英文,日语),语音情感(如 开心,悲伤)&#x…...
RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
