文档章节

NIO ByteBuffer

欧吧在线
 欧吧在线
发布于 2015/06/19 12:46
字数 1327
阅读 17
收藏 0

行业解决方案、产品招募中!想赚钱就来传!>>>

在 NIO 库中,所有数据都是用缓冲区处理的。在读取数据时,它是直接读到缓冲区中的。在 写入数据时,它是写入到缓冲区中的。任何时候访问 NIO 中的数据,都是将它放到缓冲区中。缓冲区实质上是一个数组。通常它是一个字节数组,但是也可以 使用其他种类的数组。但是一个缓冲区不仅仅是一个数组。缓冲区提供了对数据的结构化访问,而且还可以跟踪系统的读/写进程。

buffer其实只是一个美化了的数组。

状态变量
跟踪数据的状态情况使buffer可以自己管理数据资源 

position: 其实是指从buffer读取或写入buffer的下一个元素位置。比如,已经写入buffer 3个元素那那么position就是指向第4个位置,即position设置为3(数组从0开始计)。
limit:还有多少数据需要从buffer中取出,或还有多少空间可以放入。postition总是
<=limit。
capacity: 表示buffer本身底层数组的容量。limit绝不能
>capacity。
filp():作了两件事情:
1.将limit指向现在position的位置 2.将position设置为0 (limit=position;position=0
       这个过程可以使之前buffer写入数据时改变的状态变为可以“准备读取”。因为之前写到buffer中的数据就是position 到 limit
-1 两个位置之间(limit指向最后一个数据的后一个位置)。

clear():
    也作了两件事:
1. limit=capacity 2.position=0
这个过程可以使buffer读取数据时改变的状态改变为“清空并准备写入”。

访问方法
以下都以bytebuffer为例
get():
   前三个get方法是相对读取。就是相对于位置状态来读取数据,并且会改变position位置状态。
   
byte get();
   ByteBuffer get(
byte dst[]);//读取bytebuffer中数据写入 dst[]
   ByteBuffer get(byte dst[],int offset, int length);
   
   该读取数据是绝对读取(一个byte),即会忽略limit和position值。并完全绕过了缓冲区的状态统计方法。
   就是说不会改变buffer内部的位置状态。
   
byte get(int index);
 
put();
   与get类似 前四个put方法是相对读取。即受position 以及limit影响,并且会改变 position。
   ByteBuffer put( 
byte b );
   ByteBuffer put( 
byte src[] ); //从src[]写入bytebuffer
   ByteBuffer put( byte src[], int offset, int length );
   ByteBuffer put( ByteBuffer src );
   最后一个是绝对写入 不会影响position等位置状态。
   ByteBuffer put( 
int index, byte b );
 
除了byte的读写还有其他类型的读写方法。并且他们都存在相对以及绝对两类。
 
操作的典型使用:
view plaincopy to clipboardprint
?while (true{  
     buffer.clear(); 
// 准备将数据写入buffer   
     int r = fcin.read( buffer ); // channel读取外部系统的数据并写入 buffer   
     if (r==-1{  
       
break;  
     }
  
     buffer.flip(); 
//准备将数据读出buffer   
     fcout.write( buffer ); // channel读取buffer的数据并写到相应的外部系统   
}
  
while (true{
     buffer.clear(); 
// 准备将数据写入buffer
     int r = fcin.read( buffer ); // channel读取外部系统的数据并写入 buffer
     if (r==-1{
       
break;
     }

     buffer.flip(); 
//准备将数据读出buffer
     fcout.write( buffer ); // channel读取buffer的数据并写到相应的外部系统
}
 
高级应用
缓存区的分配和包装
ByteBuffer.allocate(
int);方法可以分配(创建)一个byte类型的buffer。
ByteBuffer.wrap(
byte[]);方法可以将一个已有的byte数组包装出一个新的bytebuffer对象。
后一种方式需要小心处理原来的那个byte数组。因为它可以直接访问了。
缓冲区的分片
分片就是建立“子缓冲区”。子缓冲区共享父缓冲区的一部分底层数组位置。
在某种意义上,子缓冲区就像原来的缓冲区中的一个窗口。
这样当改变子缓冲区的内容时,父缓冲区的相应位置也会被改变。
分片操作是根据当前position以及limit的值来确定的。
buffer.position( 
3 );
buffer.limit( 
7 );
ByteBuffer slice 
= buffer.slice();
只读缓冲区
asReadOnlyBuffer()方法可以返回一个与原buffer对象一样的对象,只是新的buffer对象是只读的。
直接缓冲区
sun的定义:给定一个直接字节缓冲区,Java 虚拟机将尽最大努力直接对它执行本机 I
/O 操作。也就是说,它会在每一次调用底层操作系统的本机 I/O 操作之前(或之后),尝试避免将缓冲区的内容拷贝到一个中间缓冲区中(或者从一个中间缓冲区中拷贝数据)。
创建directbuffer的方式是用ByteBuffer.allocateDirect( 
int );方法替代ByteBuffer.allocate(int);
内存影射文件I
/O
它读写要比其他IO快很多.
他使文件或文件的一部分由内存影射。但是只有操作该部分位置的数据才是以内存方式读写的,而不是整个文件读入内存。(并且他是一个os的底层机制。由os底层异步完成内存与物理磁盘上的数据同步)
影射文件可以通过FileChannel对象的map方法得到。
比如以下就是将一个文件的前1024个字节影射到内存,并创建一个MappedByteBuffer对象返回出来。MappedByteBuffer是ByteBuffer的一个子类。
MappedByteBuffer mbb 
= fc.map( FileChannel.MapMode.READ_WRITE, start, size );

欧吧在线
粉丝 0
博文 32
码字总数 4007
作品 0
沈阳
私信 提问
加载中
请先登录后再评论。
Netty那点事(三)Channel与Pipeline

Channel是理解和使用Netty的核心。Channel的涉及内容较多,这里我使用由浅入深的介绍方法。在这篇文章中,我们主要介绍Channel部分中Pipeline实现机制。为了避免枯燥,借用一下《盗梦空间》的...

黄亿华
2013/11/24
2W
22
高效率的nio框架--nio java raptor

设计初衷是提供方便易用,且高效率的nio框架,一部分实现上参考了mina。还包括线程池,编解码,内存池等机制,以便于开发高性能tcp程序。 文档后续会慢慢的补上。 整体实现上尽量少的使用锁,...

齐楠
2012/12/12
3.3K
0
Java IO 之 FileInputStream & FileOutputStream源码分析

Writer :BYSocket(泥沙砖瓦浆木匠) 微 博:BYSocket 豆 瓣:BYSocket FaceBook:BYSocket Twitter :BYSocket 一、引子 文件,作为常见的数据源。关于操作文件的字节流就是 — FileInputS...

泥沙砖瓦浆木匠
2015/10/09
855
0
GitHub上整理的一些实用工具

技术站点 • Hacker News:非常棒的针对编程的链接聚合网站 • Programming reddit:同上 • MSDN:微软相关的官方技术集中地,主要是文档类 • infoq:企业级应用,关注软件开发领域 • OS...

什么丿情况
2016/01/14
795
0
Java I/O 模型的演进

原文同步至 什么是同步?什么是异步?阻塞和非阻塞又有什么区别?本文先从 Unix 的 I/O 模型讲起,介绍了5种常见的 I/O 模型。而后再引出 Java 的 I/O 模型的演进过程,并用实例说明如何选择...

waylau
2016/03/01
4.7K
20

没有更多内容

加载失败,请刷新页面

加载更多

数据库高频面试点,事务/乐观锁/悲观锁/CAS/MySQL存储引擎

事务的ACID特性是什么? 原子性: 事务是最小的执行单位,不允许分割。事务的原子性确保动作要么全部完成,要么完全不起作用; 一致性: 执行事务前后,数据保持一致,多个事务对同一个数据读...

osc_45536bvu
23分钟前
16
0
大数据BI软件助力企业数字化转型

当下,「新基建」势头正盛,随着“新基建”成为热议话题,数字化也随之成为企业面临的新机遇和新挑战。新基建的核心就是数据,数据是数字经济和企业数字化转型的生产要素和发展动力。 再看看...

osc_0boqdoe2
25分钟前
0
0
凯旋创投来志刚:基因治疗新时代,大戏刚刚开始

  2017 年,全球第一个基因治疗方法 CAR-T 细胞药物 Kymriah 获得 FDA 上市批准,从此掀起了基因治疗的热潮。随着相关技术和政策的不断成熟,基因治疗市场也随之扩大。根据德勤发布的《引领...

osc_k3vwonkw
26分钟前
10
0
LightningChart.NET使用两个BarSeries创建简单的2D图表

本教程介绍了如何使用两个BarSeries创建简单的2D图表。 BarSeries将数据值表示为矩形条,并且可以用于以非常清晰的方式可视化数据之间的差异和方差。 在本教程中,BarSeries用于表示两年期间...

roffey
27分钟前
0
0
Mybatis trim 标签的 2 个妙用!

云栖号资讯:【点击查看更多行业资讯】 在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来! mybatis的trim标签一般用于去除sql语句中多余的and关键字,逗号,或者给sql语句前拼...

osc_x03qsedc
27分钟前
10
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部