简介

  Buffer缓冲区,首先要弄明白的是,缓冲区是怎样一个概念。它其实是缓存的一种,我们常说的缓存,包括保存在硬盘上的浏览器缓存,保存在内存中的缓存(比如Redis、memcached)。Buffer是把数据保存在内存中,它本质上用来保存数据的数据结构是数组,例如ByteBuffer是byte数组,IntBuffer是int数组等,对Buffer读写操作,其实是对该数组进行数据存放、读取、清除操作。

  Buffer可以理解成是一个容器,可以往容器里写入数据,也可以从容器中读取数据,可以一个字节一个字节的读写,也可以多个字节读写,可以在写的途中切换为读,读到一半又切换到写,等等。。。

  Buffer有以下几个属性:

  • int mark 标记
  • int position 位置
  • int limit限制
  • int capacity缓冲区的容量,Buffer的容量是固定的,在创建Buffer对象时,制定容量大小。

  这几个属性,有什么作用,用在什么地方,为何设置这4个属性。其实不用先理解,在了解reset(),flip()等方法后,就会明白这四个属性的用处。

主要方法

mark、reset

  • mark() 在Buffer当前的位置设置标记
public final Buffer mark() {
mark = position;
return this;
}
  • reset() 重置Buffer当前的位置为上一次的标记
public final Buffer reset() {
int m = mark;
if (m < 0)
throw new InvalidMarkException();
position = m;
return this;
}

  这两个方法的使用场景,从Buffer读取n个字节数据,结果读出来的数据长度比n小(比如网络是拆包问题),这时候会重新再读一次,调用reset()方法。但并不是所有读操作是从位置0开始读,所以在reset()先设置一个标记,调用mark(),最后position = mark,就是从标记的mark位置开始读。

clear,flip, rewind

  • clear() 清除Buffer
public final Buffer clear() {
position = 0;
limit = capacity;
mark = -1;
return this;
}
  • flip() 反转Buffer,何谓“反转”?在执行完写操作之后,调用调用flip()为下次的读操作准备。
public final Buffer flip() {
limit = position;
position = 0;
mark = -1;
return this;
}

  使用场景:在进行完写操作后,写入100个字节,所以当前位置position=100,为了下次读操作,读出来是这100个字节数据,所以limit也设为100,限制最多读到100个字节。下次读是从位置0开始,所以position=0。

ByteBuffer buffer = ByteBuffer.allocate(100);
String value = "zhangxh20";
buffer.put(value.getBytes());
buffer.flip();
byte [] arr = new byte[buffer.remaining()];
buffer.get(arr);
String decodeValue = new String(arr);

  看下调用flip()操作前后的对比。

  如果不做flip操作,读到的将是position到capacity之间的错误内容。

  当执行flip()操作之后,它的limit被设置为position,position设置为0,capacity不变,由于读取的内容是从position到limit之间,因此,他能够正确读取到之前写入缓冲区的内容。

  • rewind() 重绕Buffer,可以发现rewind()比flip()少了个步骤limit = positon。
public final Buffer rewind() {
position = 0;
mark = -1;
return this;
}

  使用场景:buffer.get(array),把buffer的数据复制到array数组里,会调用buffer的get()方法。get()是从position开始复制,最多limit个,所以只需positon设置为0。

其他特性

不变式

  标记、位置、限制和容量值遵守以下不变式:

  0 <= 标记 <= 位置 <= 限制 <= 容量
  新创建的缓冲区总有一个 0 位置和一个未定义的标记。初始限制可以为 0,也可以为其他值,这取决于缓冲区类型及其构建方式。一般情况下,缓冲区的初始内容是未定义的。

线程安全

  Buffer不是线程安全的,被多个线程使用时,需要加同步。

ByteBuffer

  byteBuffer字节缓冲区,即每次调用get()或者put()操作的数据最小单位是byte,理IntBuffer、LongBuffer、DoubleBuffer以此类推。例如LongBuffer put()最小的单位是8个字节的long数据。日常开发中,ByteBuffer使用的会比较多,其他类型Buffer很少使用。
  事实上ByteBuffer也可以用来表示其他类型Buffer,比如IntBuffer的get(),可以用ByteBuffer的getInt(),两个实现效果一样。

堆内存和直接内存

  Buffer在创建时,必须指定容量大小,还有分配内存的类型,有两种堆内存和直接内存,直接可以理解为堆外的内存空间,区别就是,直接内存是不受GC管理

参考资料

  • 《Netty权威指南》

Java NIO中的Buffer的更多相关文章

  1. Java NIO中的Buffer 详解

    Java NIO中的Buffer用于和NIO通道进行交互.如你所知,数据是从通道读入缓冲区,从缓冲区写入到通道中的.缓冲区本质上是一块可以写入数据,然后可以从中读取数据的内存.这块内存被包装成NIO ...

  2. java NIO中的buffer和channel

    缓冲区(Buffer):一,在 Java NIO 中负责数据的存取.缓冲区就是数组.用于存储不同数据类型的数据 根据数据类型不同(boolean 除外),提供了相应类型的缓冲区:ByteBufferC ...

  3. Java NIO中的Buffer类

    Buffer     缓冲,用于批量读写数据 Buffer是一个抽象类,基本数据类型都有实现类:XxxBuffer,比如ByteBuffer.CharBuffer.IntBuffer.DoubleBu ...

  4. Java NIO中的缓冲区Buffer(一)缓冲区基础

    什么是缓冲区(Buffer) 定义 简单地说就是一块存储区域,哈哈哈,可能太简单了,或者可以换种说法,从代码的角度来讲(可以查看JDK中Buffer.ByteBuffer.DoubleBuffer等的 ...

  5. Java I/O(3):NIO中的Buffer

    您好,我是湘王,这是我的博客园,欢迎您来,欢迎您再来- 之前在调用Channel的代码中,使用了一个名叫ByteBuffer类,它是Buffer的子类.这个叫Buffer的类是专门用来解决高速设备与低 ...

  6. Java NIO中核心组成和IO区别

    1.Java NIO核心组件 Java NIO中有很多类和组件,包括Channel,Buffer 和 Selector 构成了核心的API.其它组件如Pipe和FileLock是与三个核心组件共同使用 ...

  7. Java NIO Channel和Buffer

    Java NIO Channel和Buffer @author ixenos Channel和Buffer的关系 1.NIO速度的提高来自于所使用的结构更接近于OS执行I/O的方式:通道和缓冲器: 2 ...

  8. 转载Java NIO中的Files类的使用

    Java NIO中的Files类(java.nio.file.Files)提供了多种操作文件系统中文件的方法. Files.exists() Files.exits()方法用来检查给定的Path在文件 ...

  9. Java NIO中的缓冲区Buffer(二)创建/复制缓冲区

    创建缓冲区的方式 主要有以下两种方式创建缓冲区: 1.调用allocate方法 2.调用wrap方法 我们将以charBuffer为例,阐述各个方法的含义: allocate方法创建缓冲区 调用all ...

随机推荐

  1. python爬取斗图网中的 “最新套图”和“最新表情”

    1.分析斗图网 斗图网地址:http://www.doutula.com 网站的顶部有这两个部分: 先分析“最新套图” 发现地址栏变成了这个链接,我们在点击第二页 可见,每一页的地址栏只有后面的pag ...

  2. cmake-cmake.1-3.11.4机翻

    指数 下一个 | 上一个 | CMake » git的阶段 git的主 最新发布的 3.13 3.12 3.11.4 3.10 3.9 3.8 3.7 3.6 3.5 3.4 3.3 3.2 3.1 ...

  3. STC 单片机ADC实现原理

    模数转换器原理 数模转换器( analog to digitI converter,ADC),简称为A/D,ADC是链接模拟世界和数字世界的桥梁.它用于将连续的模拟信号转换为数字形式离散信号.典型的, ...

  4. Logistic回归 逻辑回归 练习——以2018建模校赛为数据源

    把上次建模校赛一个根据三围将女性分为四类(苹果型.梨形.报纸型.沙漏)的问题用逻辑回归实现了,包括从excel读取数据等一系列操作. Excel的格式如下:假设有r列,则前r-1列为数据,最后一列为类 ...

  5. zabbix监控历史数据清理

    2018/12/24 14:00:57  zabbix监控运行一段时间以后,会留下大量的历史监控数据,zabbix数据库一直在增大:可能会造成系统性能下降,查看历史数据室查询速度缓慢. zabbix里 ...

  6. 关于Amazon.com Seller 网络以及IP地址更换 官方回答

    Greetings from Amazon Seller Support, I understand your concern that there will be a change of IP ad ...

  7. 从零开始的Python爬虫速成指南

    序 本文主要内容:以最短的时间写一个最简单的爬虫,可以抓取论坛的帖子标题和帖子内容. 本文受众:没写过爬虫的萌新. 入门 0.准备工作 需要准备的东西: Python.scrapy.一个IDE或者随便 ...

  8. springboot 集成 swagger

    1. 首先配置swaggerConfigpackage com.lixcx.lismservice.config; import com.lixcx.lismservice.format.Custom ...

  9. Python Pygame(5)绘制基本图形

    最近很火一些简单图形构成的小游戏,这里介绍一些绘制图形的函数. 1.绘制矩形 rect(Surface,color,Rect,width=0) 第一个参数指定矩形绘制到哪个Surface对象上 第二个 ...

  10. Scrum立会报告+燃尽图(十月十六日总第七次):总结工作经验,商讨未来策略

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2197 Scrum立会master:李文涛 一.小组介绍 组长:付佳 组员 ...