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(); } }
在读取文件流时,正确姿势为:
- 向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位置。
简单随笔,以便后续查阅。