4. Java NIO Buffer

Java NIO Buffer与Channel需要配合来使用。如你所知,数据是从channel被读取至buffer,写入时则是从buffer写入到channel。buffer在本质上来说是你可以写入的一块内存区域,在这之后你也可以去读取数据。这个内存区域被包装成了NIO Buffer对象,它提供了一系列方法让使用这块内存更加容易。

Buffer说的基础用法

使用buffer去读取或写入数据,分为以下4个步骤:

将数据写入buffer
调用buffer.flip()
从buffer中读取数据
调用buffer.clear()buffer.compact()

当你向buffer中写入数据时,buffer会记录下你写入得数据数量。当你读取数据时,你需要使用flip()方法将缓冲区从写模式切换到读模式。在读模式中,buffer允许你读取所有已经写入buffer得所有数据。

在你读取所有数据后,你需要清空buffer,以便让它做好再次被写入数据的准备。有两种方式来清空buffer:调用clear()compact()。前者是清空整个缓冲区,而后者则只是清除你已经读取的数据。未读的数据则被移动到buffer的开始位置,而新数据则会被写入到未读数据的后面。

下面是一个简单的buffer应用例子:

        RandomAccessFile aFile = new RandomAccessFile("/Users/kopshome/buffer.txt", "rw");
        FileChannel inChannel = aFile.getChannel();
        // 创建buffer,capacity 设置为 48 bytes,建立文件测试时,建议字符数量大于48,这有助于你的理解
        ByteBuffer buf = ByteBuffer.allocate(48);
        int bytesRead = inChannel.read(buf); // 读取至buffer.
        while (bytesRead != -1) {
            buf.flip(); // 让buffer做好读取的准备
            while (buf.hasRemaining()) {
                System.out.print((char) buf.get()); // 每次读取1 byte
            }
            buf.clear(); //让buffer做好写入的准备
            bytesRead = inChannel.read(buf);
        }
        aFile.close();

Buffer的Capacity,Position 和 Limit

在本质上来说,buffer只是一块儿内存区域,这可以往里面写数据,也可以从中读取数据。这块儿内存区域被包装成了NIO bunffer对象,提供了一系列方法让你操作此内存区域更加方便。

Buffer含有三个你必须要熟悉的属性,以便于理解buffer是如何工作的:

  • capacity
  • position
  • limit

属性positionlimit的含义,取决于buffer当前是读模式还是写模式。capacity不管什么模式始终代表一种含义。

下面是capacity,position 和 limit在读模式和写模式下含义的示意图。具体的解释请在看完图后继续查看:

image
Capacity

作为一块儿内存区域,buffer固然要有固定的大小,也叫做“capacity”。你只能写入byte,long,字符等等。一旦buffer写满,你需要清空它(读取数据或清空)才能继续读取数据至buffer。

Position

你向buffer中写数据的过程中,写入的数据恰好是在某一确定的位置。最开始position为0。当一个byte,long等等已经被写入到buffer中,position指向的是可以写入数据的下一位置。position的极限是-1。

当你从buffer中读取数据时你也可以从一个给定的position开始读。当你从buffer的写模式切换到读模式,position会被设置回到 0。当你从buffer中读数据时时从position开始,position代表着下一个可读取的数据。

Limit

在写数模,buffer的limit代表含义是你可以将多少数据写入到buffer中。limit和capacity相等。

当buffer切换至读模式,limit的含义是你可以从buffer中读取到多少数据。所以,当将buffer切换到读模式,limit会被设置成写模式下的position。换句话说,你可以读取到所有已经写入到buffer中的数据(limit被设置成写入byte的数量,用position标记)。

buffer类型

Java NIO提供了下面几种buffer类型:

  • ByteBuffer
  • MappedByteBuffer
  • CharBuffer
  • DoubleBuffer
  • FloatBuffer
  • IntBuffer
  • LongBuffer
  • ShortBuffer

正如你所见,不同的buffer类型代表着不同的数据类型。换句话说,当你处理byte时可以用char,short,int,long,float 或 double来处理。

MappedByteBuffer有一点不同,在涉及它的章节自然会讲到。

配置buffer

想获得buffer对象,你首先得配置它。每一个Buffer类都含有一个allocate()方法。下面是一个配置ByteBuffer的例子,设置capacity为48 byte:

ByteBuffer buf = ByteBuffer.allocate(48);

下面的例子是将CharBuffer的capacity设置为1024字符长度:

CharBuffer buf = CharBuffer.allocate(1024);

向buffer中写数据

将数据写入Buffer有两种方式:

从channel种写数据到buffer
通过put()将数据写入到buffer

下面例子展示了channel如何将数据写入buffer:

int bytesRead = inChannel.read(buf); //read into buffer.

下面是利用put()方法将数据写入buffer:

buf.put(127);   

还有很多其他版本的put()方法,所以你会有很多办法把数据写入到buffer。例如在某一位置写,或将byte数组写到buffer中。更多的方法请参考Java相关api文档。

flip()

使用flip()方法可以让buffer从写模式切换至读模式。调用此方法会让position设置回0,设置limit为和position相同的值。

换句话说,position现在表示读取的位置,limit的含义是有多少字节,字符可以读取。

从buffer中读数据

从buffer中读数据有2种办法:

把数据从buffer中读取到channel
自己手动读取数据,使用get()方法

下面的例子是展示如何把数据从buffer读取到channel:

//read from buffer into channel.
int bytesWritten = inChannel.write(buf);

下面的例子是使用get()方法读取数据:

byte aByte = buf.get();   

rewind()

方法Buffer.rewind()设置position为0,所以你可以读取所有的数据。limit不受影响,依旧表示可以工buffer中读取出多少数据。

clear() h compact()

当你读取完buffer的数据之后,你必须让buffer做好被写入下次数据的准备。你可以使用clear() ``或者``compact()完成此操作。

如果你调用了clear()方法,position被设置成了0。换句话说,buffer被清空了。buffer中的数据并没有被清空,只是告诉你哪里可以写到数据缓冲区。

如果buffer中存在没有被读取出的数据,然后你调用了clear()方法,那么这写数据就会被"forgotten",意思就是你再也不会知道哪些数据已经被读取,哪些数据没有被读取了。

如果buffer中有未被读取的数据,然后你想稍后再读,那么你可以调用compact()而不是clear()方法。

方法compact()会把数据拷贝到buffer的开始。然后它设置position为下一个未读数据的位置。limit依然和capacity相等。现在buffer已经准备好被写入数据了,但是不不会覆盖未读数据。

mark() 和 reset()

你可在buffer中调用mark()方法来设置一个给定的position。稍后你可以使用buffer.reset()方法来重置position至上一步用mark()方法标记的位置。下面是具体的例子:

buffer.mark();
//call buffer.get() a couple of times, e.g. during parsing.
buffer.reset(); //set position back to mark. 

equals() 和 compareTo()

使用这两个方法可以变焦两个buffer。

equals()

两个buffer是相等的,如果满足以下条件:

  • 它们是同一类型(byte, char, int 等等)
  • 它们有相同数量的剩余字节,字符
  • 剩下的字节,字符相通

如你所见,这只比较buffer的一部分,并不是比较每一个里面的元素。实际上这只比较buffer的剩余部分。

compareTo()

此方法比较两个buffer剩余的元素,使用场景包括排序程序等等。一个buffer被认为比另一个buffer"smaller",应给满足以下条件:

  • 第一个不相等的元素小于另一个Buffer中对应的元素 。
  • 所有元素都相等,但第一个Buffer比另一个先耗尽(第一个Buffer的元素个数比另一个元素个数少)。

想要查看此教程的目录请点击:Java NIO教程目录贴地址

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,817评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,329评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,354评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,498评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,600评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,829评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,979评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,722评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,189评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,519评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,654评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,329评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,940评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,762评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,993评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,382评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,543评论 2 349

推荐阅读更多精彩内容