掌握ByteBuffer与CharBuffer需遵循写入、flip、读取流程,通过allocate或allocateDirect创建缓冲区,利用put、get操作数据,flip切换模式,clear或compact复用缓冲区,结合CharsetDecoder/Encoder处理字符编码,正确管理position、limit与capacity是高效I/O处理的关键。

在Java中处理I/O数据时,ByteBuffer 和 CharBuffer 是NIO(New I/O)包中的核心类,常用于高效地操作原始字节和字符数据。它们通过缓冲机制减少系统调用次数,提升性能。理解如何正确使用这些Buffer类,对开发高性能网络应用或文件处理程序至关重要。
创建与分配缓冲区
使用 ByteBuffer.allocate() 可以创建固定大小的堆内存缓冲区,适合常规场景。若需直接内存以提升I/O效率,可用 allocateDirect()。
- ByteBuffer byteBuf = ByteBuffer.allocate(1024);- CharBuffer charBuf = CharBuffer.allocate(512);
CharBuffer 通常由解码器从ByteBuffer生成,不建议手动填充字符数据。
写入与读取数据的基本流程
所有Buffer都遵循“写模式 → 翻转 → 读模式”的操作顺序。关键方法包括 put()、flip()、get() 和 clear() 或 compact()。
立即学习“Java免费学习笔记(深入)”;
- 写入字节:byteBuf.put("Hello".getBytes());- 切换为读模式:byteBuf.flip();
- 读取数据:while (byteBuf.hasRemaining()) System.out.print((char)byteBuf.get());
flip() 方法重置 position 并设置 limit,使读取能从头开始直到之前写入的末尾。
使用编码器与解码器转换字节与字符
直接操作CharBuffer较少见,通常通过 CharsetDecoder 将ByteBuffer转为字符流。
- CharsetDecoder decoder = StandardCharsets.UTF_8.newDecoder();- CharBuffer charBuf = decoder.decode(byteBuf);
反之,使用 CharsetEncoder 可将字符串编码回ByteBuffer。这种方式避免乱码,确保字符集一致性。
清除与复用缓冲区
处理完一批数据后,调用 clear() 重置缓冲区,适用于完全消费后的场景。若缓冲区中仍有未读数据,应使用 compact() 将剩余数据移至开头,便于下次读取。
- byteBuf.clear(); // 重置position,limit=capacity- byteBuf.compact(); // 保留未读部分,后续可继续填充
这是实现非阻塞I/O循环读写的关键技巧。
基本上就这些。掌握ByteBuffer和CharBuffer的核心操作逻辑,配合标准编码工具,能有效处理大多数缓冲数据场景。关键是理解position、limit、capacity三者的关系,以及flip和clear的语义差异。不复杂但容易忽略细节。










