Java NIO Buffer

  • buffer的基本使用
  • buffer的capacity、position、limit属性
  • buffer类型
  • 分配缓存区
  • 向buffer中写入数据
  • flip()方法
  • 从buffer中读取数据
  • rewind()方法
  • clear()和compact()方法
  • mark()和reset()方法
  • equal()和compareTo()方法

  Java buffer 和 channel 是相辅相成的,正如前文所言,通过channel写入数据到buffer中,通过channel从buffer中读取数据。

  buffer是一个高校的内存块,供你写入数据用。这些内存块在buffer中被包装起来了,并且提供了方法让你轻松使用内存块。

   Basic Buffer Usage

  用buffer进行读取或写入数据,通常有如下四个步骤:

  1. 向buffer中写入数据
  2. 调用buffer的flip()方法
  3. 从buffer中读取出数据
  4. 调用buffer的clear()或者compact()方法

  当你向buffer中写入数据,buffer会保存你向其中写入了多少个byte。一旦你从buffer中读取数据,你需要将buffer从写模式转化为读模式,通过flip()方法。

  一旦你读取到所有的data后,你需要清空缓存区,让buffer为下一次写入做准备。你可以通过两种方法,clear()和compact()。clear()方法会清空所有的缓存,compact()方法仅仅清空你已经读过的数据。没有读过的数据会被转移到buffer的开始处,会在没有读过的数据后写入新数据。

  以下是一个简单的例子

  



1 RandomAccessFile aFile = new RandomAccessFile("data/nio-data.txt", "rw");
 2 FileChannel inChannel = aFile.getChannel();
 3 
 4 //create buffer with capacity of 48 bytes
 5 ByteBuffer buf = ByteBuffer.allocate(48);
 6 
 7 int bytesRead = inChannel.read(buf); //read into buffer.
 8 while (bytesRead != -1) {
 9 
10   buf.flip();  //make buffer ready for read
11 
12   while(buf.hasRemaining()){
13       System.out.print((char) buf.get()); // read 1 byte at a time
14   }
15 
16   buf.clear(); //make buffer ready for writing
17   bytesRead = inChannel.read(buf);
18 }
19 aFile.close();



 Buffer Capacity、Position、Limit

  position和limit的含义会随着buffer的模式转换而不同,capacity无论哪一种模式,含义都是相同的。放两张图上来

                    

java buffer 读取字节_数据

  Capacity,在创建一个缓存区时,缓存区的大小是固定的。你只能向缓存区中写入capacity的数据量,一旦缓存区满了,你需要在下一次写入新的数据时,通过clear()方法清空缓存区。

  position,当你想缓存区写入数据时,你需要清楚你写到了哪个位置。position初始化为0,写入多少byte,position就会移动多少。position的最大值是capacity-1。

  当你从缓存区读取数据时,要从给定的position处开始读。当你使用flip()方法,将buffer的写模式转换为读模式。position会被初始化为0,当你从buffer中读取出数据时,position会提前移到下一个要读的位置。

  limit,在写模式中,limit就代表着你可以向buffer中写入多少数据,在写模式中limit等于capacity。

  当转换buffer为读模式时,limit意味着你可以读取多少数据。因此,当buffer转换为读模式时,limit会被赋值为写模式中的position。另一种是,你写进去多少数据就读多少数据(limit被赋值为写入buffer中的字节数)

 Buffer Types

  Java NIO 有以下几种buffer类型

  1. ByteBuffer
  2. CharBuffer
  3. ShortBuffer
  4. IntBuffer
  5. LongBuffer
  6. FloatBuffer
  7. DoubleBuffer
  8. MappedByteBuffer

  正如你所看见的那样,不同的buffer类型代表了不同的基本数据类型。

  MappedByteBuffer是一种特殊的buffer类型,之后再讨论它。

 Allocating a Buffer

  在获得buffer之前,你必须进行分配。每一个Buffer类都有allocate()方法来进行分配,这里有一个例子,展示如何分配内存



ByteBuffer buf = ByteBuffer.allocate(48);



 

 Writing Data to a Buffer

  你可以通过两种方法向buffer写入数据

  1. 从channel向buffer写入数据
  2. 通过buffer自己的put()方法,向buffer写入数据

  例子:



1 int bytesRead = inChannel.read(buf); //read into buffer.



 



1 buf.put(127);



 

这里还有很多put()方法的其他版本,允许你通过不同的方法向buffer中写入数据。

 flip()

  flip()方法将buffer从写模式转化为读模式。作用看代码:



1     public final Buffer flip() {
2         limit = position;
3         position = 0;
4         mark = -1;
5         return this;
6     }



 

 Reading Data From a Buffer

  这里有两种从buffer中读取数据的方法

  1. 通过channel读取数据
  2. 调用buffer自己的get()方法,读取数据

  例子:



1 //read from buffer into channel.
2 int bytesWritten = inChannel.write(buf);



1 byte aByte = buf.get();



 

  同样,也有很多版本的get()方法供你使用。

 rewind() 

  先贴上代码:



1     public final Buffer rewind() {
2         position = 0;
3         mark = -1;
4         return this;
5     }



  将position赋值为0,其余的不做变化。可以从头开始读取数据。

 clear()和compact()

  一旦你从buffer中读取完数据,你必须让buffer准备好下一次写入。你可以调用clear()方法和compact()方法。

  clear方法,会将position设置为0,还是上源码吧



1     public final Buffer clear() {
2         position = 0;
3         limit = capacity;
4         mark = -1;
5         return this;
6     }



 

  源码的注释说明:实际上clear()方法并不会将数据从缓存区中擦除,只是名字搞的鬼,因为大多数情况他确实会被用类清除数据

  compact()方法,会将你没有读完的数据复制到buffer的开始处,然后将position设置为未读数据的下一个位置。

 mark()和reset()

  在buffer中你可以调用mark()方法标记一个给定的position位置,之后你可以通过调用reset()方法,重新设置position到刚刚mark()的位置。可以自己试验下



1 buffer.mark();
2 
3 //call buffer.get() a couple of times, e.g. during parsing.
4 
5 buffer.reset();  //set position back to mark.



 

 equals()和compareTo()

  两个比较buffer的方法

  equals(),当满足以下条件时,为true

  1. 他们是相同的类型的buffer
  2. 他们有相同的剩余字节量
  3. 所有剩余的数据都相等

  正如你所看到的,equals()只能判断buffer的一部分数据。直接上源码



1     public boolean equals(Object ob) {
 2         if (this == ob)
 3             return true;
 4         if (!(ob instanceof ByteBuffer))
 5             return false;
 6         ByteBuffer that = (ByteBuffer)ob;
 7         if (this.remaining() != that.remaining())
 8             return false;
 9         int p = this.position();
10         for (int i = this.limit() - 1, j = that.limit() - 1; i >= p; i--, j--)
11             if (!equals(this.get(i), that.get(j)))
12                 return false;
13         return true;
14     }



 

  compareTo(),直接上源码了



1     public int compareTo(ByteBuffer that) {
2         int n = this.position() + Math.min(this.remaining(), that.remaining());
3         for (int i = this.position(), j = that.position(); i < n; i++, j++) {
4             int cmp = compare(this.get(i), that.get(j));
5             if (cmp != 0)
6                 return cmp;
7         }
8         return this.remaining() - that.remaining();
9     }