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读取和写入数据通常遵循这四骤:

  1. 将数据写入缓冲区
  2. 调用 buffer.flip()
  3. 将数据从缓冲区读出
  4. 调用 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

  当你从缓存中读取数据时,你也可以从一个给定的位置开始读取。当你改变缓冲区模式从写入模式到读取模式时,位置是从你所读数据的位置重置回。类似于缓冲区读取数据这样Bufferposition 前进到下一个位置进行阅读。

  限制

  在写入模式下,缓冲区的限制是可以写入缓冲区的数据量的限制,限制等于该缓冲区的容量

  在缓冲区的读取模式下,限制是可以从数据中读取多少数据的限制。因此,当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);

将数据写入缓冲区

  您可以通过两种方式将数据写入缓冲区:

    1. 将数据从Channel写入Buffer
    2. 通过缓冲区的put()方法自己写入数据。

  以下是一个示例,显示如何Channel将数据写入Buffer

int bytesRead = inChannel.read(buf); //读入缓冲区

这是一个Buffer通过该put()方法将数据写入到数据的示例:

buf.put(127);

  put()方法允许以许多不同的方式写入数据到缓冲区。例如,在特定位置写入,或者将一个字节数组写入缓冲区等等。

flip()   模式切换

  flip()方法可以切换缓冲区的写入模式到读取模式。调用flip()position返回设置为0,并将其设置limit 为刚刚的位置。

  position现在标记了读取位置,并limit标记了被写入缓冲区多少字节,字符等等(也就是可以读取多少字节,字符等的限制)。

从缓冲区读取数据

  从缓冲区读取数据的两种方式。

  1. 将数据从缓冲区读入通道。
  2. 从缓冲区中读取数据,使用其中的一个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.所有元素是相等的,但是第一个缓冲区在第二个缓冲区之前用完了元素(它具有较少的元素)。

JAVA NIO学习笔记二 频道和缓冲区的更多相关文章

  1. Java NIO 学习笔记(二)----聚集和分散,通道到通道

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

  2. Java NIO 学习笔记(三)----Selector

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

  3. Java NIO 学习笔记(一)----概述,Channel/Buffer

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

  4. Java NIO 学习笔记(七)----NIO/IO 的对比和总结

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

  5. Java NIO 学习笔记(五)----路径、文件和管道 Path/Files/Pipe

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

  6. Java NIO 学习笔记(四)----文件通道和网络通道

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

  7. Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer

    作者:Grey 原文地址:Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer ByteBuffer.allocate()与ByteBuffer.allocateD ...

  8. Java IO学习笔记二

    Java IO学习笔记二 流的概念 在程序中所有的数据都是以流的方式进行传输或保存的,程序需要数据的时候要使用输入流读取数据,而当程序需要将一些数据保存起来的时候,就要使用输出流完成. 程序中的输入输 ...

  9. Java NIO学习笔记

    Java NIO学习笔记 一 基本概念 IO 是主存和外部设备 ( 硬盘.终端和网络等 ) 拷贝数据的过程. IO 是操作系统的底层功能实现,底层通过 I/O 指令进行完成. 所有语言运行时系统提供执 ...

随机推荐

  1. markdown在typora的超方便的应用

    经常性需要进行文字的编写,但是使用word比较麻烦,而excel又不方便自己的工作实际实用,特此有一款软件支持我的应用,typora软件,让人使用的无法自拨. typora是基于markdown的来应 ...

  2. css中 禁止spa有点击状态

    <span class="an" onclick="selNum();"></span> var selNum = function() ...

  3. elk with docker-compose

    version: '2' services: elasticsearch: image: docker.calix.local:18080/docker-elasticsearch:6.2.2-1 # ...

  4. POJ 2893 M × N Puzzle——八数码有解条件

    题意:给定M*N的数码图,问能否移动到最终状态 分析 有解的判定条件可见 八数码有解条件 值得一提的是,这道题求逆序对卡树状数组,只能用归并排序. #include<cstdio> #in ...

  5. 快捷键IntelliJ IDEA For Mac

    http://www.cnblogs.com/wxd0108/p/5295017.html Mac键盘符号和修饰键说明 ⌘ Command ⇧ Shift ⌥ Option ⌃ Control ↩︎  ...

  6. 12、生命周期-@Bean指定初始化和销毁方法

    12.生命周期-@Bean指定初始化和销毁方法 Bean的生命周期:创建->初始化->销毁 容器管理bean的生命周期 我们可以自定义初始方法和销毁方法,容器在bean进行到当期那生命周期 ...

  7. CF1156E Special Segments of Permutation

    思路:笛卡尔树?(好像并不一定要建出来,但是可以更好理解) 提交:2次 错因:没有判左右儿子是否为空来回溯导致它T了 题解: 建出笛卡尔树,考虑如何计算答案: 先预处理每一个值出现的位置 \(pos[ ...

  8. [BJWC2008]王之财宝

    嘟嘟嘟 如果没有限制,而且必须选\(m\)件的话,就是隔板法\(C_{n + m - 1} ^ {m - 1}\)了.现在要选至多\(m\)件,那么就相当于新增一个板儿,分出的新的盒子表示" ...

  9. DP的优化

    参考资料: 李煜东<算法竞赛进阶指南> 斜率优化 形如: \(f[i] = min\{f[j]+val(i,j)\}\)的dp,多项式\(val(i,j)\)包含\(i,j\)的乘积项 引 ...

  10. RNN(二)——基于tensorflow的LSTM的实现

    lstm的前向结构,不迭代 最基本的lstm结构.不涉及损失值和bp过程 import tensorflow as tf import numpy as np inputs = tf.placehol ...