JavaByteBufferflip与clear使用解析
时间:2026-04-10 11:58:33 173浏览 收藏
Java ByteBuffer的flip()和clear()并非简单的“切换模式”或“清空数据”,而是精准操控其内部position、limit等元数据的状态机操作:flip()是写转读的关键步骤,它将当前写入位置设为读取边界并重置游标,确保读操作覆盖已写内容;clear()则用于重置为可写状态,但不擦除底层数据,仅指针归位;而compact()适用于需保留未读数据的流式场景。理解这些方法的本质——它们只改变逻辑视图而非物理内存,且与堆内/堆外实现无关——才能避免BufferUnderflowException、BufferOverflowException等典型错误,真正驾驭ByteBuffer作为高效、可控的字节缓冲区的核心机制。

ByteBuffer写完后读不到数据?一定是没调用flip()
写完数据后直接get()或get(byte[])返回0或抛BufferUnderflowException,本质是position没重置、limit没收缩。写模式下limit等于capacity,position停在写入末尾;不flip(),读操作就从“末尾”开始读,自然读不到东西。
正确流程只有两步:put() → flip() → get()。其中flip()干三件事:把当前position设为新的limit(即有效数据长度),把position归零,clear mark(如果有的话)。
flip()后position=0,limit=原position,正好覆盖已写区域- 多次
flip()无害,但反复flip()→get()→flip()会导致读取范围不断缩小(因为每次get()移动position,下次flip()的limit就变小) - 如果写入中途中断(比如只put了3个字节),
flip()仍以当前position为准,不会自动补零或截断
想重复写入新数据,该用clear()还是compact()?
clear()是重置缓冲区最常用的操作,但它不丢数据——只是把position设为0、limit设为capacity、mark设为-1。底层array()或堆外内存里的字节全还在,只是后续put()会从头覆盖。
真正要保留未读完的数据并追加写,得用compact():它把position到limit之间的未读数据移到buffer开头,然后position设为已复制长度,limit设为capacity。适合“边读边写”的流式场景(比如网络收包后处理一部分,剩下留着和下次数据拼接)。
clear():适合“一次写完→读完→清空→再写”的循环,比如HTTP响应体构造compact():适合“读一部分→处理→剩余数据暂存→等下次数据来拼接”,比如TCP粘包处理rewind()不能代替clear():它只重置position=0,limit不变,无法释放写空间
为什么flip()之后再put()会抛BufferOverflowException?
因为flip()后limit被设为之前写入的长度,此时buffer逻辑上是“只读状态”。再调put()就会尝试往[0, limit)区间写,但position从0开始递增,很快超过limit。
典型误操作:写→flip→读一部分→想继续写新数据→直接put。这时必须先clear()(丢弃所有旧数据)或compact()(保留未读部分),才能恢复写能力。
- 错误链:
put(3); flip(); get(1); put(1);→ 第二个put()触发异常 - 修复方式取决于意图:想丢弃全部就
clear();想保留未读的2字节就compact()(此时buffer前2字节是原数据,position=2,可继续put) - 没有“读写双工”模式:ByteBuffer始终是单向状态机,读写切换必须靠
flip()/clear()/compact()
堆内vs堆外ByteBuffer对flip()/clear()有影响吗?
没有。这两个方法只操作buffer的四个元数据字段(mark、position、limit、capacity),和底层内存是否在堆内无关。堆外buffer(ByteBuffer.allocateDirect())执行flip()一样快,也不触发GC。
但要注意:堆外内存的生命周期不由JVM自动管理,即使buffer对象被回收,内存可能滞留。不过这和flip()无关,属于Cleaner机制范畴。
- 所有ByteBuffer子类(heap/direct/mapped)都继承同一套状态逻辑
- 唯一影响性能的点是:direct buffer的
get()/put()涉及JNI调用,比heap buffer慢,但flip()/clear()本身开销可忽略 - 别在循环里反复
allocateDirect()+clear(),容易触发系统级内存分配失败,应复用buffer
本篇关于《JavaByteBufferflip与clear使用解析》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
334 收藏
-
420 收藏
-
320 收藏
-
396 收藏
-
263 收藏
-
325 收藏
-
490 收藏
-
260 收藏
-
125 收藏
-
400 收藏
-
118 收藏
-
430 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习