Java NIO教程 Buffer
缓冲区本质上是一块可以写入数据,然后可以从中读取数据的内存,这块内存中有很多可以存储byte(或int、char等)的小单元。这块内存被包装成NIO Buffer对象,并提供了一组方法,用来方便的访问该块内存。
为了理解Buffer的工作原理,需要熟悉它的三个属性:
- capacity
- position
- limit
简单的解释这三个属性的含义可以概括为:capacity代表这块Buffer的容量,position代表下一次读(或写)的位置,limit代表本次读(或写)的极限位置。这么简单说一下你当然是听不懂的啦(这样一下你就听懂了,岂不是显得我很没有存在感)所以下面开始详细的讲解。
capacity
作为一个内存块,Buffer有一个固定的大小值(这个大小是刚开始申请的),叫作“capacity”.你只能往里写capacity个byte、int,char等类型。
position
当你要写数据到Buffer中时,position表示当前可写的位置。初始的position值为0(也可以用过方法进行改变)。当一个byte、int等数据写到Buffer后, position会向前移动到下一个可插入数据的Buffer单元。position最大可为capacity – 1.
当读取数据时,也是从某个特定位置读。当从Buffer的position处读取数据完成时,position向前移动到下一个可读的位置。
limit
在写数据时,Buffer的limit表示你最多能往Buffer里写多少数据,position移动到limit写操作停止。初始limit的值等于Buffer的capacity。当读取数据时, limit表示你最多能读到多少数据,position移动到limit读操作停止。
无论在读数据时还是在写数据时,只要position超过了limit就会抛出异常。
控制position和limit的值
capacity的值是根据申请Buffer的大小和种类确定的,所以不能改变。而position和limit就可以根据我的需要而改变了,首先介绍一下如何查看这三个属性的值:buffer.limit()
、buffer.position()
、buffer.capacity()
这三个方法直观、方便,我们就不再罗嗦。
接下来我们要着重看一下buffer.flip()
这个方法一般用在写到读切换的时候。这个方法的能力就是将limit设为position的值,再是将position设为0。
这么做的用处是什么呢?你想想,在写数据的时候从Buffer的开始处—0位置到position位置之间已经写满了数据,如果这时候我们想要从头开始读数据的话,就要将position指向0,以便可以读取0位置的数据,然后逐个向下读取;但要读到什么位置为止呢?如果整个Buffer都读完的话,刚才所写的最后一个单元以后的单元,都是空,读取它们没有意义。所以读取到刚才所写的最后一个单元,是明智之举。而在将position置为0之前,position值就是刚才所写的最后一个单元的位置。所以在写到读切换的时候,将limit设为position的值,再是将position设为0。
这个明白了以后一切都顺了。buffer.clear()
是清空Buffer的方法,但它没有真正的清除,只是将position置为0,将limit置为capacity;这样一来,你的写操作就可以将原来的数据覆盖了。就是这么简单。buffer.rewind()
是将position置为0,这样一来就可以将buffer再重新读一遍,当然你还可通过它干很多事。
其实还有很多有用、有趣的方法,看看api文档吧。
基础实例 和 btye与其他类型的转换
说了这么多理论,再不上代码就有人得骂街了,来个最基础的申请buffer和基本的读写吧
ByteBuffer bb = ByteBuffer.allocate(48);
/*向ByteBuffer中put数据的时候,一下四种形式都可以
* put(byte b)
* put(byte[] src)
* put(byte[] src, int offset, int length)
* put(ByteBuffer src)
* 四种形式都会移动position指针
*/
bb.put(new byte[]{1,2,4,2,-13});
bb.flip();
//hasRemaining()的作用是看看position到没到limit位置
while(bb.hasRemaining()) {
System.out.println(bb.get());
}
还得来一段理论,再听我扯一会。Buffer共有类型有以下几种
- ByteBuffer
- CharBuffer
- DoubleBuffer
- FloatBuffer
- IntBuffer
- LongBuffer
- ShortBuffer
- MappedByteBuffer(这哥们儿有点特殊,以后单独再讲)
Buffer在nio中的主要作用就是与channel交互。但这几种类型中能与channel交互的只有ByteBuffer(坑爹的吧!)所以在用其他类型Buffer的时候,一般都是先将ByteBuffer转化为想用的类型,用的是byteBuffer.asCharBuffer()
、byteBuffer.asIntBuffer()
等方法进行转换。这种方式用术语来讲,叫做“产生其他种类Buffer的视图”;意思就是底层是ByteBuffer,但看起来是其它种类的Buffer,可以用相应的方法,但是视图发生了读写,底层的ByteBuffer也会发生变化。
下面这段是将ByteBuffer转化为CharBuffer视图的例子
ByteBuffer bb = ByteBuffer.allocate(1024);
//将ByteBuffer转化为CharBuffer视图后,再调用put,ByteBuffer中的position指针不会移动
bb.asCharBuffer().put("Hello World");
//为了能正确的输出,这里改变了limit指针的位置,使之变到了字符数组的末尾
bb.limit("Hello World".length()*Character.BYTES);//字符数组长度*每个字符占的字节数
while(bb.hasRemaining()) {
System.out.print(bb.getChar());
}
/*也可用如下的方法输出
* while((c=bb.getChar())!=0) {
* System.out.print(c);
* }
*/
这段例子告诉我们,视图发生了读写,底层的ByteBuffer是有感知的,以及感知如何展现出来。但真正用的时候,没这么蛮烦。因为学了后面的channel,就知道了,channel直接就把整个ByteBuffer都拿走了,就不用这样一个个的输出了。而且一个个输出的话也可以直接利用视图层,向下看
ByteBuffer bb = ByteBuffer.allocate(1024);
IntBuffer ib = bb.asIntBuffer();
ib.put(new int[]{1,42,12,-12});
/*将ByteBuffer转化为IntBuffer视图后,再调用put,ByteBuffer中的position指针不会移动
* 但是所生成的IntBuffer中的position会按正常方式移动
* 而且整个IntBuffer的capacity会按照byte 和 int 之间的所占字节大小比例而改变*/
System.out.println("ByteBuffer.position = "+bb.position());
System.out.println("ByteBuffer.limit = "+bb.limit());
System.out.println("ByteBuffer.capacity = "+bb.capacity());
System.out.println("IntBuffer.position = "+ib.position());
System.out.println("IntBuffer.limit = "+ib.limit());
System.out.println("IntBuffer.capacity = "+ib.capacity());
ib.flip();
while(ib.hasRemaining()) {
System.out.println(ib.get());
}
会了这些Buffer的知识就差不多了,就到这里了。多打打例子代码、多体会体会,就可以洗洗睡了,拜拜
还是那句话,有问题及时告诉我
Java NIO教程 Buffer的更多相关文章
- Java NIO教程 目录
"Java NIO系列教程" 是笔者hans为NIO的初学者编写的一份入门教程,想仔细学习的同学可以按照顺序去阅读.由于我学的也不是特别的精,所以错误.疏漏在所难免,希望同学们指正 ...
- Java NIO 之 Buffer(缓冲区)
一 Buffer(缓冲区)介绍 Java NIO Buffers用于和NIO Channel交互. 我们从Channel中读取数据到buffers里,从Buffer把数据写入到Channels. Bu ...
- Java NIO 之 Buffer
Java NIO 之 Buffer Java NIO (Non Blocking IO 或者 New IO)是一种非阻塞IO的实现.NIO通过Channel.Buffer.Selector几个组件的协 ...
- Java NIO之Buffer(缓冲区)
Java NIO中的缓存区(Buffer)用于和通道(Channel)进行交互.数据是从通道读入缓冲区,从缓冲区写入到通道中的. 缓冲区本质上是一块可以写入数据,然后可以从中读取数据的内存.这 ...
- 海纳百川而来的一篇相当全面的Java NIO教程
目录 零.NIO包 一.Java NIO Channel通道 Channel的实现(Channel Implementations) Channel的基础示例(Basic Channel Exampl ...
- Java NIO 教程
Java NIO(New IO)是从Java 1.4版本开始引入的一个新的IO API,可以替代标准的Java IO API.本系列教程将有助于你学习和理解Java NIO. Java NIO提供了与 ...
- Java NIO之Buffer的使用
目录 Buffer简介 Buffer的核心属性 Buffer的创建与使用(ByteBuffer为例) 总结 参考资料 Buffer简介 缓冲区(Buffer):本质上是一个数组,用于临时保存.写入以及 ...
- JAVA NIO简介-- Buffer、Channel、Charset 、直接缓冲区、分散和聚集、文件锁
IO 是主存和外部设备 ( 硬盘.终端和网络等 ) 拷贝数据的过程. IO 是操作系统的底层功能实现,底层通过 I/O 指令进行完成. Java标准io回顾 在Java1.4之前的I/O系统中,提供 ...
- Java NIO教程 前言
阅读本文前,建议你先了解 旧I/O NIO 是 New I/O 的缩写,要了解它真正的内涵,需要掌握的知识还是比较多的.我努力在这几篇笔记里,勾勒出整个io的面貌.为大家的深入学习铺路. I/O简史 ...
随机推荐
- CDH自己安装方式Hbase master备份方式
hbase-daemon.sh start master 启动改命令会默认产生Hmaster 进程,等待主机宕机,zookeepr 监控
- uva 10375
/* 选择与除法_________________________________________________________________________________ #include & ...
- (转)The AlphaGo Replication Wiki
The AlphaGo Replication Wiki 摘自:https://github.com/Rochester-NRT/RocAlphaGo/wiki/01.-Home Contents : ...
- Mabitis 多表查询(一)resultType=“java.util.hashMap”
1.进行单表查询的时候,xml标签的写法如下 进行多表查询,且无确定返回类型时 xml标签写法如下: <select id="Volume" parameterType=&q ...
- 子类实例化和Super
在子类的构造函数当中,必须调用父类的构造函数,通过super的参数个数和类型来决定调用父类哪一个构造函数. class Student extends Person{ Student(){ super ...
- volley_之基本使用
Volley的基本使用-------------- 本人初学,如有纰缪,望指正~ Volley是Google在2003年的I/O大会上推出的通信框架,结合了AsyncHttpClient和Univer ...
- CRM Xrm.Page 的对象层次结构
- chrome设置可以跨域访问
右键chrome的快捷方式->属性 修改目标属性:添加--args --disable-web-security --user-data-dir=F:\MyChromeDevUserData, ...
- 2013年第四届蓝桥杯C/C++程序设计本科B组决赛
1.猜灯谜(枚举) 2.连续奇数和(等差数列) 3.空白格式化(去除空格) 4.高僧斗法(阶梯nim) 5.格子刷油漆(dp) 6.农场阳光 1.猜灯谜 A 村的元宵节灯会上有一迷题:请猜谜 * 请猜 ...
- cocos2d-x + Lua接入iOS原生SDK的实现方案[转]
相信很多朋友在使用cocos2d-x+lua开发游戏时都遇到过接入iOS原生SDK的问题,比如常见的接应用内支付SDK,广告SDK或是一些社交平台SDK等等,我也没少接过这类SDK.这篇文章主要是对我 ...