Java NIO-Buffer

Buffer和Channel总是成对出现,在Java NIO中Buffer用于和NIO通道进行交互,数据总是从Chanel中读入缓冲区,然后在从缓冲区写入Channel中。

缓冲区本质上是一块物理上连续分配的内存区域,这块内存区域被包装成NIO Buffer对象,并提供了一组方法,用于方便的访问该内存区域。Buffer映射操作能够直接操作底层平台的资源。这些操作节省了在不同地址空间中复制数据的开销——这在现代计算机体系结构中是开销很大的操作(相比于Java 面向流的IO)。

1- 常用的Buffer类型


可以通过char,short,int,long,float 或 double类型来操作缓冲区中的字节, MappedByteBuffer,用于表示内存映射文件。

2- Buffer的分配

在使用Buffer之前需要先分配指定大小的内存区域。分配的缓冲区大小是定长的,不可以扩展容量。并将分配的缓冲区元素都置为0。

  1. 不同类型Buffer的分配

每个Buffer类型的数据都有一个allocate的静态方法来分配指定类型大小的缓冲数据区域,而且这些Buffer类型都是抽象类,不可实例化(但是可以使用类的静态方法)。

  1. ByteBuffer不同内存区域的Buffer分配
  • 在Java堆上分配内存,HeapByteBuffer是NIO的包内访问权限类,包外不可获取,方法返回向上转型为ByteBuffer,其他Buffer类也都有这个方法,分配的是堆上内存区域(新建了一个byte数组)。

  • 在堆外内存上分配缓冲区,只有ByteBuffer可以分配堆外缓冲区。

    堆外内存使用unsafe.allocateMemory方法分配

    但是可以先分配ByteBuffer的堆外内存,然后转换为其他类型的来访问。

    还有堆外分配不一定能分配成功,需要进行判断
  1. 堆上(HeapByteBuffer) VS 堆外(DirectByteBuffer)
  • 分配和销毁堆外直接内存缓冲区通常要比分配和销毁堆上缓冲区消耗更多的系统资源。
  • DirectByteBuffer比HeapByteBuffer读写性能更高,可以提高网络交互的速度:HeapByteBuffer会发生频繁的直接内存和JVM堆内存之间的相互的拷贝,比如flush数据到远程的时候,会将JVM内存拷贝到直接内存然后才会进行数据发送的工作;而DirectByteBuffer是对直接内存的保证所以会省去内存拷贝的过程,这在计算操作系统中节省可观的性能消耗。
  • 基于NIO的开源web框架如Netty,Nginx都基于DirectByteBuffer提高了整体读写性能。如Netty的ByteBuf完成了对ByteBuffer的包装和拓展,需要注意的是在NIO网络编程时只有ByteBuffer能和Channel进行读写交互,ByteBuf在于Channel进行交互时也是转换成了ByteBuffer之后再与Channel交互。
  • 由于分配DirectByteBuffer比较消耗系统资源,但是又能提高读写性能,Netty的做法是分配DirectByteBuffer后自己根据引用计数做内存回收,重新复用DirectByteBuffer,而不是直接释放掉分配好的内存
  1. DirectByteBuffer回收管理(程序员完全控制管理)

直接内存的释放是GC(full gc,调用System.gc())自动回收来控制的,并不由程序员控制,没有类似的close或者free等显示释放内存空间的方法,但是如何才能有程序员完全掌握回收的控制权呢?DirectByteBuffer有一个Cleaner域用于内存释放,给了程序猿一线生机。

  1. 设置JVM 参数DisableExplicitGC ,禁用full gc(这样DirectByteBuffer就不会被系统回收了) ,严重警告:禁用full gc,需要严格的测试,存在内存泄露的风险,必要进行堆外内存管理(其实Netty就是这么干的)
  2. DirectByteBuffer构造函数会新建一个Deallocator类来初始化这个Cleaner域
  1. Deallocator是DirectByteBuffer静态内部类,含有一个run方法,方法内部使用unsafe.freeMemory释放分配的直接内存空间


  2. Cleaner类有一个方法clean方法,调用的是传进去的Deallocator对象的run方法,可以用来释放分配的堆外直接内存。


  3. 代码示例

    DirectByteBuffer实现了DirectBuffer接口,DirectByteBuffer是default访问权限,但是DirectBuffer是public,如果不是出于特殊考虑建议不要通过DirectBuffer直接操作DirectByteBuffer,容易造成安全隐患(这也是DirectByteBuffer定义为default访问权限的原因)

3- Buffer的使用

Buffer的使用一般搭配Channel

  1. 将数据读到Buffer中
  1. 将Buffer中数据读出

get方法也可以获取指定位置的数据

  1. position(位置)、capacity(容量)、limit(限制)

position指的是当前在缓冲数组中的位置;capacity指的是缓冲数组的大小,在创建时指定,代表着最大可存储的数据长度;limit指的是有效数据的长度,limit小于等于capacity。

  1. 读写模式转换时,PCL的变化
  • 新建:position=0;limit = capacity
  • 读入数据大小5byte:position = 5;limit=capacity
  • flip(写->读)模式转换:position=0;limit=转换前position位置
  • clear方法:和新建一样
  • rewind方法:position=0;limit=保持不变

flip方法用于写->读转换,clear方法用于重置缓冲数组等待下一次将数据写入缓冲数组,rewind方法用于重读数组,需要注意的是这些只是position、limit 的位置在发生变化,缓冲数组的数据并没有被清除,只有下次写入才能将原来的数据覆盖。当将缓冲数组数据读出时,只会读出position-limit范围内的数据(有效数据),而不会读取limit-capacity之间的数据(上次写入时遗留的数据,等待被覆盖)
其他的方法还有mark()与reset()方法,通过调用Buffer.mark()方法,可以标记Buffer中的一个特定position。之后可以通过调用Buffer.reset()方法恢复到这个position。这样就提供了在一个缓冲数组中反复遍历操作读入数据的便利和灵活性,而不像面向流的IO不能操作当前位置的前一个字节。

参考
http://ifeve.com/buffers/

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

推荐阅读更多精彩内容