1. 缓冲区Buffer
NIO中的buffer是一个用于存储基本数据类型值的容器。在使用传统I/O流API时,如InputStream和OutputStream,我们常常把字节流的数据放入byte[]字节数组中,或放入char[]字符数组中进行读写操作,这种处理方式处理比较原始。NIO中为我们提供了缓冲区Buffer类来解决,它还有很多其他方法。
1.1 认识Buffer
这个类小伙伴们平时开发中应该经常使用到,可能你还并不是特别了解它。下面我们来仔细看看,如下图所示:
首选Buffer是一个抽象类,它有7个子类,分别是:IntBuffer、FloatBuffer、CharBuffer、DoubleBuffer、ShortBuffer、LongBuffer、ByteBuffer。
注意
- 8大基本数据类型它又对应的7中,除了boolean类型,BooleanBuffer这个类并不存在;
- StringBuffer这个类是java.lang包下面的,Buffer是在java.nio包下面。
1.2 Buffer属性
Buffer又四个重要的属性,如下所示:
// Invariants: mark <= position <= limit <= capacity
private int mark = -1;
private int position = 0;
private int limit;
private int capacity;
- mark:标记,代表标记索引,
- position:位置,代表缓冲区下一个要读取或写入元素的下标;
- limit:限制,代表缓冲区第一个不能被读取或写入元素的下标;
- capacity:容量,代表缓冲区最大的容量,在创建时设置。
他们的大小关系:
0<= mark<=position<=limit<=capacity
打个形象的比喻:高铁线路从北京到深圳,你要到,mark表示你买的武汉站,position表示高铁即将停靠的下一站,limit表示武汉站的下一站,不能坐到的站,因为武汉站是你的下车站。而capacity表示高铁的终点站。
这样是不是很好记忆啦。
2. Buffer的方法
2.1 构造方法
Buffer(int mark, int pos, int lim, int cap) { // package-private
if (cap < 0)
throw new IllegalArgumentException("Negative capacity: " + cap);
this.capacity = cap;
limit(lim);
position(pos);
if (mark >= 0) {
if (mark > pos)
throw new IllegalArgumentException("mark > position: ("
+ mark + " > " + pos + ")");
this.mark = mark;
}
}
2.2 获取位置与设置
// 获取position
public final int position() {
return position;
}
// 设置position
public final Buffer position(int newPosition) {
if ((newPosition > limit) || (newPosition < 0))
throw new IllegalArgumentException();
position = newPosition;
// 当mark>position时,mark被丢弃,设置为-1
if (mark > position) mark = -1;
return this;
}
2.3 reset方法
//reset方法将mark的值赋给position
public final Buffer reset() {
int m = mark;
if (m < 0) //如果mark小于0则抛出异常
throw new InvalidMarkException();
position = m;
return this;
}
2.4 还原缓冲区状态
clear()方法作用是将缓冲区还原为初始状态,即“恢复出厂设置”。将位置设置为0,将限制设置为容量,并丢弃标记。它主要用在即将对缓冲区进行存储数据之前调用该方法。
public final Buffer clear() {
position = 0;
limit = capacity;
mark = -1;
return this;
}
需要特别注意的是:该方法随叫clear,**实际上它不能真正清除缓冲区的数据。**当执行该方法后再进行read操作,新数据会覆盖旧数据。
2.5 对缓冲区进行反转
file方法作用是:缩小limit范围,将限制limit复制为当前的位置position,position设置为0,mark丢弃。该方法常应用在向缓冲区写入一些数据后,然后下一步进行读取之前调用,这样就可以重头开始读取数据了。
public final Buffer flip() {
limit = position;
position = 0;
mark = -1;
return this;
}
2.6 重新读取缓冲区
rewind方法作用是:将位置posittion设置为0,丢弃标记mark,限制limit不变。rewind英文含义为重绕,倒带,即该方法是重新读取缓冲区数据的意思。
public final Buffer rewind() {
position = 0;
mark = -1;
return this;
}
总结一下:
- clear方法重点是还原缓冲区状态;
- flip方法重点是截取,类似substring;
- rewind方法重点是重新读取缓冲区数据。