Java NIO 频道
Java NIO渠道类似于流,他们之间具有一些区别的:
- 您可以读取和写入频道。流通常是单向(读或写)。
- 通道可以异步读取和写入数据。
- 通道常常是读取或写入缓冲区。
如上所述,您将数据从通道读入缓冲区,并将数据从缓冲区写入通道。以下是流程图:
Java NIO:Channels read data into Buffers, and Buffers write data into Channels |
Channel 的实现
以下是Java NIO中最重要的Channel实现:
- FileChannel
- DatagramChannel
- SocketChannel
- ServerSocketChannel
FileChannel--------
从文件中读取和读入数据。
DatagramChannel----
可以通过UDP网络读取和写入数据。
SocketChannel------
可以通过TCP网络读取和写入数据。
ServerSocketChannel---
让您监听进入的TCP连接,例如Web服务器一样。对于每个传入连接 SocketChannel
都创建。
简单的Channel示例
这是使用一个FileChannel
将一些数据读取到一个Buffer的简单示例
:
RandomAccessFile aFile = new RandomAccessFile(“data / nio-data.txt”,“rw”); FileChannel inChannel = aFile.getChannel(); ByteBuffer buf = ByteBuffer.allocate(48); int bytesRead = inChannel.read(buf); while(bytesRead!= -1){ System.out.println(“Read”+ bytesRead); buf.flip(); while(buf.hasRemaining()){ System.out.print((char)buf.get()); } buf.clear(); bytesRead = inChannel.read(buf); } aFile.close();
buf.flip()
通话。首先你读入缓冲区。然后你翻转它,然后再把它读出来。
Java NIO缓冲区
与NIO通道交互时,使用Java NIO缓冲区。数据从通道读入缓冲区,并从缓冲区写入通道。
缓冲区本质上是一个可以写入数据的内存块,然后可以再次读取。该内存块被包装在一个NIO缓冲区对象中,它提供了一组方法,使其更容易使用内存块。
基本缓冲区使用
使用a Buffer
读取和写入数据通常遵循这四骤:
- 将数据写入缓冲区
- 调用
buffer.flip()
- 将数据从缓冲区读出
- 调用
buffer.clear()
或buffer.compact()
当数据写入缓冲区时,缓冲区就会对数据的写入量进行跟踪。一旦需要读取数据,您需要使用flip()
方法调用将缓冲区从写入模式切换到读取模式。在读取模式下,缓冲区可以读取写入缓冲区的所有数据。
当你读完所有的数据时,你需要清除缓冲区,使其为再次写入做准备。你可以通过两种方法来清楚数据:通过调用clear()
或 compact()方法
。clear()
方法清除整个缓冲区。compact()
方法仅清除您已经读取的数据。任何未读的数据都被移动到缓冲区的开头,数据将在未读数据之后被写入缓冲区。
这是一个简单的Buffer
用法示例,用粗体字写入,翻转,读取和清除操作:
RandomAccessFile aFile = new RandomAccessFile(“data / nio-data.txt”,“rw”);FileChannel inChannel = aFile.getChannel();//创建容量为48字节的缓冲区ByteBuffer buf = ByteBuffer.allocate(48);int bytesRead = inChannel.read(buf); //读入缓冲区while(bytesRead!= -1){ buf.flip(); //使缓冲区准备好读取 而(buf.hasRemaining()){ System.out.print((char)buf.get()); //一次读取1个字节 } buf.clear(); // make buffer ready for writing bytesRead = inChannel.read(buf);}aFile.close();
缓冲区的容量、位置、和限制
缓冲区有三个重要属性:
- 容量
- 位置
- 限制
位置
和限制的含义
取决于缓冲区的是读模式还是写的模式。不管是缓冲区是模式,容量含义是相同的。
以下是对写入和读取模式的容量,位置和限制的说明。
写入和读取模式下的缓冲区容量,位置和限制。 |
容量
缓冲区具有一定的大小,也称为其“容量”。您只能将capacity
字节,长整型,字符等写入缓冲区。如果缓冲区已满,您需要清空写入的数据(读取数据或清除它),然后再写入数据。
位置
当你想缓冲区写入数据时,你在某个位置上这样做。最初的位置为0。当一个字节,长整型等已经被写入Buffer,
位置被提前指向缓冲区中的下一个单元格以便插入数据。位置可以最大程度地成为 capacity - 1
。
当你从缓存中读取数据时,你也可以从一个给定的位置开始读取。当你改变缓冲区模式从写入模式到读取模式时,位置是从你所读数据的位置重置回0。类似于缓冲区读取数据这样Buffer
,position
前进到下一个位置进行阅读。
限制
在写入模式下,缓冲区的限制是可以写入缓冲区的数据量的限制,限制等于该缓冲区的容量。
在缓冲区的读取模式下,限制是可以从数据中读取多少数据的限制。因此,当Buffer
转换到读取模式时,将限制设置为写入模式的写入位置。换句话说,您可以读取与写入一样多的字节(限制设置为写入的字节数,以位置标记)。
缓冲区类型
Java NIO缓冲区类型如下:
- ByteBuffer
- MappedByteBuffer
- CharBuffer
- DoubleBuffer
- FloatBuffer
- IntBuffer
- LongBuffer
- ShortBuffer
您可以看到,这些Buffer
类型表示不同的数据类型。也就是,它们可以处理缓冲区中的字节为char,short,int,long,float或double。
这MappedByteBuffer
有点特别。
分配缓冲区
要获取Buffer
对象,您必须先分配它。每个Buffer
类都有一个allocate()
的分配方法。这是一个ByteBuffer分配实例, ByteBuffer
容量为48个字节:
ByteBuffer buf = ByteBuffer.allocate(48);
这是一个CharBuffer
为1024个字符分配示例:
CharBuffer buf = CharBuffer.allocate(1024);
将数据写入缓冲区
您可以通过两种方式将数据写入缓冲区:
-
- 将数据从
Channel
写入Buffer
- 通过缓冲区的
put()
方法自己写入数据。
- 将数据从
以下是一个示例,显示如何Channel
将数据写入Buffer
:
int bytesRead = inChannel.read(buf); //读入缓冲区
这是一个Buffer
通过该put()
方法将数据写入到数据的示例:
buf.put(127);
put()
方法允许以许多不同的方式写入数据到缓冲区。例如,在特定位置写入,或者将一个字节数组写入缓冲区等等。
flip() 模式切换
flip()
方法可以切换缓冲区的写入模式到读取模式。调用flip()
将position
返回设置为0,并将其设置limit
为刚刚的位置。
position
现在标记了读取位置,并limit
标记了被写入缓冲区多少字节,字符等等(也就是可以读取多少字节,字符等的限制)。
从缓冲区读取数据
从缓冲区读取数据的两种方式。
- 将数据从缓冲区读入通道。
- 从缓冲区中读取数据,使用其中的一个get()。
以下是一个示例,说明如何将数据从缓冲区读取到通道中:
//从缓冲区读入通道。int bytesWritten = inChannel.write(buf);
以下是Buffer
使用get()方法读取数据的示例:
byte aByte = buf.get();
get()
方法允许从Buffer以
许多不同的方式读取数据 。例如,在特定位置读取,或从缓冲区读取字节数组。
byte aByte = buf.get();
rewind()
缓冲区的rewind()方法,将
positio返回设置
为0,因此您可以重新读取缓冲区中的所有数据。在限制
保持不变,因此仍然标记多少个元素(字节,字符等),可以从缓冲区被读取。
clear()和compact()
一旦你完成阅读数据之后,你必须设置Buffer
为下一次写入做准备。你可以通过打调用clear()
或调用compact()来完成上述操作
。
如果调用clear()
的position
是重新设置返回为0,limit设置成容量
。或者说是,缓冲区
被清除,缓冲区的数据未被清除。只有标记从哪里可以将数据写入缓冲区
。
如果Buffer
您在调用clear()
该数据时有未读的数据将被“忘记”,这意味着您不再有任何标记可以告诉您读取了哪些数据,还那些没有被读取。
如果还有未读的数据Buffer
,并且您想稍后阅读,但您需要先写一些写,调用compact()
而不是调用clear()
。
compact()
将所有未读取的数据复制到开头Buffer
。然后它设置position
在最后一个未读元素之后。限制大小仍然设置容量,就像clear()
。现在Buffer
已经为做好准备,但是不会覆盖未读的数据。
mark()和reset()
可以通过调用缓冲区的mark()
方法来标记给定的位置。然后,您可以稍后通过调用该缓冲区的reset()
方法将位置重置回标记的位置。这是一个例子:
buffer.mark();
//调用buffer.get()几次,例如在解析过程中。buffer.reset(); //将位置设置为标记。
equals()和compareTo()
可以使用equals()
和compareTo()比较两个缓冲区。
equals()
两个缓冲区相等的条件:
A.它们是相同类型(byte,char,int等)
B.它们在缓冲区中具有相同数量的剩余字节,字符等。
C.所有剩余的字节,字符等等。
综上所述,equals只比较缓冲区其中的一部分,而不是每个元素。其实,它只是比较缓冲区的剩余元素。
compareTo()
compareTo()
方法比较两个缓冲区的其余元素(字节,字符等),以便在例如排序线程中使用。
缓冲区相比其他缓冲区,“小”的条件:
A.与其他缓冲区中的相应元素相等的第一个元素小于另一个缓冲区中的元素。
B.所有元素是相等的,但是第一个缓冲区在第二个缓冲区之前用完了元素(它具有较少的元素)。