ByteBuffer中的flip()、clear()、compact()

简介: ByteBuffer中的flip()、clear()、compact()

NIO读取文件流的简单demo:

public static void main(String[] args) {
        try (FileChannel channel = new FileInputStream("data.txt").getChannel()){
            // 定义缓冲区 allocate分配大小
            ByteBuffer buffer = ByteBuffer.allocate(10);
            while(true) {
                // 去读缓冲区内容
                int read = channel.read(buffer);
                if(read != -1){
                    // 切换为读模式
                    buffer.flip();
                    while (buffer.hasRemaining()) {
                        System.out.println((char)buffer.get());
                    }
                    // 切换为写模式
//                    buffer.clear();
                    buffer.compact();
                } else {
                    break;
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

image.gif

在读取文件流时,正确姿势为:

    • 向buffer写数据,如channel.read(buffer),此时默认为写模式
    • 调用flip()切换为读模式
    • 从buffer读取数据,如buffer.get()
    • 调用clear()或compact()切换为写模式
    • 重复以上步骤

    为什么需要频繁切换读写模式呢?这里简单介绍一下flip(),clear(),compact()三者到底做了什么事。

    1、ByteBuffer的内部结构

    首先介绍三个属性:

    capacity:bytebuffer容量大小
    position:位移指针位置
    limit:指针位移限制

    image.gif

    image.png

    ByteBuffer buffer = ByteBuffer.allocate(10)分配得到的初始空间大小为:

    capacity:10字节

    position:0

    limit:为capacity的位置

    当有数据写入时候,每写入一个字节,position往后移动一个字节位置,直到limit位置,该缓冲区满。

    image.png

    2、flip()方法

    如需要读取缓冲区数据,使用flip()切换为读数据模式,此时position和limit指针位置发生变化。

    image.png

    position移动到最开始位置,limit移动到数据长度的末尾。每读取一个字节,position向后移动一个字节位置。直到limit位置处停止。

    3、clear()方法

    调用clear()方法,position和limit指针位置发生变化。

    clear默认清空缓冲区,回到最开始分配的样子,即position=0,limit=capacity。

    image.png

    4、compact()方法

    compact也可以切换为写模式,不过是把未读完成的数据向前压缩,然后再切换。

    image.png

    如果缓冲区中数据没有读取完,就需要立马写入数据,所以需要对未读取完的数据进行压缩,避免数据丢失。compact也会改变position和limit指针位置,但position != 0,而是等于上次读取的位置,limit=capacity位置。

    简单随笔,以便后续查阅。

    相关文章
    |
    1月前
    |
    Java
    让星星⭐月亮告诉你,Java NIO之Buffer详解 属性capacity/position/limit/mark 方法put(X)/get()/flip()/compact()/clear()
    这段代码演示了Java NIO中`ByteBuffer`的基本操作,包括分配、写入、翻转、读取、压缩和清空缓冲区。通过示例展示了`position`、`limit`和`mark`属性的变化过程,帮助理解缓冲区的工作原理。
    28 2
    nobuffer与av_read_frame的关系
    nobuffer与av_read_frame的关系
    80 0
    |
    算法
    frame_size (1536) was not respected for a non-last frame
    frame_size (1536) was not respected for a non-last frame
    100 0
    frame_size (1536) was not respected for a non-last frame
    |
    网络协议
    av_interleaved_write_frame -32 broken pipe
    av_interleaved_write_frame -32 broken pipe
    258 0
    av_interleaved_write_frame -32 broken pipe
    |
    编解码
    av_read_frame返回值为-5 Input/output error
    av_read_frame返回值为-5 Input/output error
    161 0
    |
    Java Linux Android开发
    NoSuchMethod: ByteBuffer.position(I)
    NoSuchMethod: ByteBuffer.position(I)
    93 0
    |
    存储 架构师 关系型数据库
    double write buffer,你居然没听过?
    double write buffer,不是一个内存buffer,是一个内存/磁盘两层的结构,是InnoDB里On-Disk架构里很重要的一部分; 是一个通过写两次,保证页完整性的机制。
    555 0
    double write buffer,你居然没听过?
    |
    存储 SQL 缓存
    写缓冲(change buffer),这次彻底懂了!!!
    一次了解缓冲池(buffer pool)。
    588 0
    写缓冲(change buffer),这次彻底懂了!!!