缓存区 Buffer 是数据容器

  • ByteBuffer 可以存储除了 boolean 以外的其他 7 种Java基本数据类型,如 getInt、putInt
  • Buffer 是抽象类,它有除了 Boolean 以外的其他 7 种Java基本数据类型子类,如IntBuffer

缓存区的初始化

⑴ 静态方法 allocate:初始化一个指定容量的缓存区

解释一下图例:

1.数组下标:数字(1,2,3,4,5)表示数组下标

1.数组:灰色长方形内表示真实有效的数组。

2.数组元素:数字所在的白色黑边正方形表示数组元素,可以通过例如 array[0] 来获取数组元素。但是注意,图中没有明确指定数组元素的值是什么。

3.虚拟下标:其中 -16 是虚拟的下标位置,如果使用这2个下标取值( array[-1] 或者 array[6] )会抛出 IndexOutOfBoundsException

⑵ 静态方法 wrap:把一个数组初始化成一个缓存区

  • 如果是只把数组 array 作为 wrap 的参数,那么效果和 allocate 类似。但是,修改数组中的值,等同于修改缓存区的值。
  • 如果除了数组引用 array 外,还有 offsetlength 来指定子数组的开始位置和子数组的大小。效果如下图:

  • 灰色的部分就是数组参数 array
  • 虚线框的部分表示从数组中划出的 子数组

缓存区存储数据

  • put() 填充一个新的元素到数组中
  • 当前位置右移(position++)

缓冲区获取数据

  • get() 获取当前位置的元素
  • 当前位置右移(position++)

核心操作

flip 操作

由写模式 转换为> 读模式

  • limit = position,限制转到原先的 position 位置
  • position = 0,游标归零
  • mark = -1,标记清空

注意:

虽然,flip 的中文含义是 “翻动”,但是连续调用 flip 不能实现 写模式 -> 读模式 -> 写模式!

连续调用 flip 之后,postion == limit == 0。

因此,继续调用 get() 会抛出 读取超限 BufferUnderflowException ,调用 put() 会抛出 写入超限 BufferOverflowException

clear 操作

重新回到初始化状态,重新进入写模式

  • limit = capacity,限制回归到容器容量
  • position = 0,游标归零
  • mark = -1,标记清空

注意

clear 实际并不会清理数据,而是调整 limit、position、mark 指向的位置

mark 操作

  • mark = position,记录下当前位置

使用场景

替换一段内容:添加标记,以便后续调用 reset() 将 position 回到标记。

reset 操作

  • position = mark,当前位置回到标记位置

注意

  • mark < 0 抛出无效标记异常。说明必须要先 mark() 才能调用 reset()

rewind 操作

  • position = 0,当前位置归零
  • mark = -1,清除标记位置

剩余空间与超限异常

剩余空间

\[0 <= 标记 <= 位置 <= 限制 <= 容量
\]

其中,

\[remaining = limit - position
\]

remaining 表示剩余空间大小。hasRemaining() 用来查询是否还有剩余空间。

读取超限

  • get 读取数据时,如果 position >= limit,抛出 BufferUnderflowException 异常

写入超限

  • put 写入数据时,如果 position >= limit,抛出 BufferOverflowException 异常

注意

如果是使用 wrap 初始化的缓冲区,即使没有到达容器容量 capacity,也会报错,因为此时 limit < capacity

总结:

  • Buffer缓冲区是数据容器,可以用于读 get() 也可以用于写 put()。
  • flip() 可以从写模式切换到读模式
  • clear() 可以回到初始化状态
  • mark() 和 reset() 需要配合使用,主要用于 替换某段内容
  • 无论是读取超限还是写入超限,条件都是 position >= limit。即 如果进行接下来的读或者写操作,当前位置都将超过限制。

NIO入门之缓冲区Buffer的更多相关文章

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

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

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

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

  3. 详解 缓冲区(Buffer 抽象类)

    在本篇博文中,本人主要讲解NIO 的两个核心点 -- 缓冲区(Buffer) 和 通道 (Channel)之一的 缓冲区(Buffer), 有关NIO流的其他知识点请观看本人博文<详解 NIO流 ...

  4. Java NIO入门(二):缓冲区内部细节

    Java NIO 入门(二)缓冲区内部细节 概述 本文将介绍 NIO 中两个重要的缓冲区组件:状态变量和访问方法 (accessor). 状态变量是前一文中提到的"内部统计机制"的 ...

  5. NIO(一)——缓冲区Buffer

                                        NIO(一)--Buffer NIO简介 NIO即New IO,是用来代替标准IO的,提供了与标准IO完全不同传输方式. 核心: ...

  6. Java NIO -- 缓冲区(Buffer)的数据存取

    缓冲区(Buffer): 一个用于特定基本数据类型的容器.由 java.nio 包定义的,所有缓冲区都是 Buffer 抽象类的子类.Java NIO 中的 Buffer 主要用于与 NIO 通道进行 ...

  7. NIO之缓冲区(Buffer)的数据存取

    缓冲区(Buffer) 一个用于特定基本数据类行的容器.有java.nio包定义的,所有缓冲区都是抽象类Buffer的子类. Java NIO中的Buffer主要用于与NIO通道进行交互,数据是从通道 ...

  8. JAVA NIO缓冲区(Buffer)------ByteBuffer常用方法

    参考:https://blog.csdn.net/xialong_927/article/details/81044759 缓冲区(Buffer)就是在内存中预留指定大小的存储空间用来对输入/输出(I ...

  9. Java NIO 缓冲区 Buffer

    缓冲区 Buffer 是 Java NIO 中一个核心概念,它是一个线性结构,容量有限,存放原始类型数据(boolean 除外)的容器. 1. Buffer 中可以存放的数据类型 java.nio.B ...

随机推荐

  1. ElasticSearch--validate验证搜索语句是否合法或者存在语法错误

    GET /accounts/person/_validate/query?explain { "query":{ "match": { "user&q ...

  2. Java并发编程:Callable、Future和FutureTask 实现龟兔赛跑

    1.不清楚的看博客http://www.cnblogs.com/dolphin0520/p/3949310.html 我们使用上面的代码来实现一个龟兔赛跑 package com.weiyuan.te ...

  3. springboot自动装配原理

    最近开始学习spring源码,看各种文章的时候看到了springboot自动装配实现原理.用自己的话简单概括下. 首先打开一个基本的springboot项目,点进去@SpringBootApplica ...

  4. ceph集成openstack cinder

    本环境ceph已经搭建,ceph搭建麻烦见本博客的其他文章 1 在cinder-volume节点安装ceph client yum install -y ceph-common 注意:glance要安 ...

  5. javaScript的三种储存方式

    (一).SessionStorage     会话储存 (二).localStorage           本地储存 (三).Cookier                   现实中为:饼干    ...

  6. 手摸手带你理解Vue的Watch原理

    前言 watch 是由用户定义的数据监听,当监听的属性发生改变就会触发回调,这项配置在业务中是很常用.在面试时,也是必问知识点,一般会用作和 computed 进行比较. 那么本文就来带大家从源码理解 ...

  7. hive如何获取当前时间

    在大多数的sql中获取当前时间都是用now()函数即可,hive获取当前时间的函数与sql 不一样 在impala中执行now()函数时是可以通过的 然而在hive中执行now()函数却报错: hiv ...

  8. Blazor带我重玩前端(二)

    概览 Blazor目前有两种托管模式,一种是Server-Side模式,一种是WebAssembly模式.官方首先支持的是Service-Side模式,使用WebAssembly模式,需要更新到最新版 ...

  9. Azure Web App (一)发布你的Net Core Web 项目

    一,引言 今天我们看一下Azure上的一个服务-----Web 应用,我们都知道云计算的三大模式:Iaas(基础设施即服务),Paas(平台即服务),Saas(软件即服务). Iass,其实就是虚拟主 ...

  10. ELKF-分布式日志收集分析平台搭建 最小化 配置过程 - 查看收集日志(windows10下搭建)

    前言 Elasticsearch是与名为Logstash的数据收集和日志解析引擎以及名为Kibana的分析和可视化平台一起开发的.这三个产品被设计成一个集成解决方案,称为“Elastic Stack” ...