JAVA NIO学习笔记二 频道和缓冲区
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。
当你从缓存中读取数据时,你也可以从一个给定的位置开始读取。当你改变缓冲区模式从写入模式到读取模式时,位置是从你所读数据的位置重置回。类似于缓冲区读取数据这样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.所有元素是相等的,但是第一个缓冲区在第二个缓冲区之前用完了元素(它具有较少的元素)。
JAVA NIO学习笔记二 频道和缓冲区的更多相关文章
- Java NIO 学习笔记(二)----聚集和分散,通道到通道
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java NIO 学习笔记(三)----Selector
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java NIO 学习笔记(一)----概述,Channel/Buffer
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java NIO 学习笔记(七)----NIO/IO 的对比和总结
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java NIO 学习笔记(五)----路径、文件和管道 Path/Files/Pipe
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java NIO 学习笔记(四)----文件通道和网络通道
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer
作者:Grey 原文地址:Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer ByteBuffer.allocate()与ByteBuffer.allocateD ...
- Java IO学习笔记二
Java IO学习笔记二 流的概念 在程序中所有的数据都是以流的方式进行传输或保存的,程序需要数据的时候要使用输入流读取数据,而当程序需要将一些数据保存起来的时候,就要使用输出流完成. 程序中的输入输 ...
- Java NIO学习笔记
Java NIO学习笔记 一 基本概念 IO 是主存和外部设备 ( 硬盘.终端和网络等 ) 拷贝数据的过程. IO 是操作系统的底层功能实现,底层通过 I/O 指令进行完成. 所有语言运行时系统提供执 ...
随机推荐
- linux下setsockopt函数的使用
1.closesocket(一般不会立即关闭而经历TIME_WAIT的过程)后想继续重用该socket:BOOL bReuseaddr=TRUE;setsockopt(s,SOL_SOCKET ,SO ...
- python logger日志配置
self.logger = logging.getLogger(logName) # 创建logger实例 time = datetime.datetime.now() logFilePath = o ...
- ZZNU-OJ-2098 : Drink coffee【线段树合并区间或者 差分 + 二分索引树】
: Drink coffee 时间限制: Sec 内存限制: MiB 提交: 答案正确: 提交 状态 讨论区 题目描述 为了在上课时保持清醒,凯伦需要一些咖啡.咖啡爱好者凯伦想知道最佳的温度来冲煮完美 ...
- 洛谷1546 最短网络Agri-Net【最小生成树】【prim】
[内含最小生成树Prim模板] 题目:https://www.luogu.org/problemnew/show/P1546 题意:给定一个邻接矩阵.求最小生成树. 思路:点少边多用Prim. Pri ...
- springboot2.0入门(九)-- springboot使用mybatis-generator自动代码生成
一.配置文件引入 插件引入,引入 <plugin> <groupId>org.mybatis.generator</groupId> <artifactId& ...
- MySQL中的连接、实例、会话、数据库、线程之间的关系
MySQL中的实例.数据库关系简介 1.MySQL是单进程多线程(而Oracle等是多进程),也就是说MySQL实例在系 统上表现就是一个服务进程,即进程(通过多种方法可以创建多实例,再安装一个端口号 ...
- HDU 6143 - Killer Names | 2017 Multi-University Training Contest 8
/* HDU 6143 - Killer Names [ DP ] | 2017 Multi-University Training Contest 8 题意: m个字母组成两个长为n的序列,两序列中 ...
- scapy2 爬取全站,以及使用post请求
前情提要: 一:scrapy 爬取妹子网 全站 知识点: scrapy回调函数的使用 二: scrapy的各个组件之间的关系解析 Scrapy 框架 Scrapy是用纯Python实现一个为了爬取网 ...
- Vue-es6基础语法
什么是ES6 ECMAScript 6 简称ES6, 在2015年6月正式发布~ ECMAScript 是JavaScript语言的国际标准. 我们本着二八原则,掌握好常用的,有用的~能让我们更快的 ...
- 网络1911、1912 C语言第4次作业--函数批改总结
目录 网络1911.1912 C语言第4次作业--函数批改总结 一.评分规则 二.本次作业亮点 三.本次作业存在的问题 四.助教有话说 五.得分详情 网络1911.1912 C语言第4次作业--函数批 ...

