前言
本篇博文是《从0到1学习 Netty》系列的第一篇博文,主要内容是介绍 NIO 的核心之一 Buffer 中的 ByteBuffer,文章中所提到的代码都在我的 GitHub 仓库中,点击此处跳转;
什么是 Netty?
Netty 是一个高性能、异步事件驱动的网络应用程序框架,主要用于快速开发可维护、可扩展的高性能服务器和客户端。Netty 提供了简单易用的 API,支持多种协议和传输方式,并且有着高度灵活的扩展和自定义能力。
Netty 的设计目标是提供一种易于使用、高效、可扩展的异步 IO 网络编程框架。它采用了 NIO(Non-blocking IO)的方式来进行网络操作,避免了传统的阻塞式 IO 常常面临的性能瓶颈。同时,Netty 还提供了优秀的线程模型和内存管理机制,保证了高并发下的稳定性和性能。
通过 Netty,开发者可以方便地实现基于 TCP、UDP、HTTP、WebSocket 等多种协议的通信应用。同时,Netty 还提供了编解码器、SSL 支持等组件,使得开发者可以更加专注于业务逻辑的实现。
什么是 ByteBuffer?
ByteBuffer 是 Java 中的一个类,它提供了一种方便的方式来处理原始字节数据。ByteBuffer 可以被看作是一个缓冲区,它可以容纳一定数量的字节数据,并提供了一系列方法来操作这些数据。
使用 ByteBuffer,可以轻松地读取和写入二进制数据。它还提供了对不同类型数据的支持,如整数、浮点数等。ByteBuffer 还支持对数据进行切片,以及对缓冲区中的数据进行复制、压缩、解压等操作。
在 Java 中,ByteBuffer 通常用于处理 I/O 操作,例如从文件或网络中读取和写入数据。它也可以用于处理加密和解密数据,以及处理图像和音频文件等二进制数据。总之,ByteBuffer 是 Java 中非常有用的一个类,可以帮助开发人员更轻松地处理二进制数据。
基本使用
-
向 buffer 写入数据,例如调用
channel.read(buffer)
; -
调用
flip()
切换至读模式flip
会使得 buffer 中的 limit 变为 position,position 变为 0
-
从 buffer 读取数据,例如调用
buffer.get()
; -
调用
clear()
或者compact()
切换至写模式- 调用
clear()
方法时,position=0
,limit 变为 capacity; - 调用
compact()
方法时,会将缓冲区中的未读数据压缩到缓冲区前面;
- 调用
-
重复 1~4 的步骤;
编写代码进行测试:
@Slf4j
public class TestByteBuffer {
public static void main(String[] args) {
try (FileChannel channel = new FileInputStream("data.txt").getChannel()) {
// 准备缓冲区
ByteBuffer buffer = ByteBuffer.allocate(10);
while (true) {
// 从 channel 读取数据写入到 buffer
int len = channel.read(buffer);
log.debug("读取到的字节数 {}", len);
if (len == -1) break;
// 打印 buffer 内容
buffer.flip(); // 切换至读模式
while(buffer.hasRemaining()) { // 是否还有剩余未读数据
byte b = buffer.get();
log.debug("实际字节 {}", (char)b);
}
buffer.clear();
}
} catch (IOException e) {
}
}
}
复制代码
运行结果:
注意,日志需要进行配置,在 /src/main/resources/
路径下,创建 logback.xml
,由于篇幅原因,自行在我的 GitHub 上下载:logback.xml;
将末尾部分的 <logger name="com.sidiot.netty" level="DEBUG" additivity="false">
中的 name
的属性值改成自己的包名即可。
部分读者可能会遇到如下问题:
这是由于 lombok
引起的,需要检查一下是否安装了 lombok
的插件,以及是否是最新版的 lombok
,博主这里用的版本如下:
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.26</version>
</dependency>
复制代码
内部结构
字节缓冲区的父类 Buffer