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

发布时间 2023-12-10 14:31:45作者: ck_2016
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();
        }
    }
View Code

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

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


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

1、ByteBuffer的内部结构

首先介绍三个属性:

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

ByteBuffer buffer = ByteBuffer.allocate(10)分配得到的初始空间大小为:
capacity:10字节
position:0
limit:为capacity的位置
当有数据写入时候,每写入一个字节,position往后移动一个字节位置,直到limit位置,该缓冲区满。

 2、flip()方法

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

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

3、clear()方法

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

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

4、compact()方法

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

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


 

 

 

转发:https://blog.csdn.net/p793049488/article/details/122523894