目录

  • Buffer简介
  • Buffer的核心属性
  • Buffer的创建与使用(ByteBuffer为例)
  • 总结
  • 参考资料

Buffer简介

缓冲区(Buffer):本质上是一个数组,用于临时保存、写入以及读取数据。在Java NIO中,

该内存块包含在NIO Buffer对象当中,NIO Buffer对象还提供了一组接口来访问该内存块。

根据数据类型的不同,Java为除了boolean类型之外的其余7种基本类型提供了相应类型的缓冲区,

分别是ByteBufferCharBufferShortBufferIntBufferLongBuffer

FloatBufferDoubleBuffer。他们都继承自抽象类Buffer类,他们的管理方式也都几乎一样。

UML类图如下:

Buffer的核心属性

BUffer类的部分实现如下:

public abstract class Buffer {
// Invariants: mark <= position <= limit <= capacity
private int mark = -1;
private int position = 0;
private int limit;
private int capacity; //构造方法
Buffer(int mark, int pos, int lim, int cap) { // package-private
if (cap < 0)
throw new IllegalArgumentException("Negative capacity: " + cap);
this.capacity = cap;
limit(lim);
position(pos);
if (mark >= 0) {
if (mark > pos)
throw new IllegalArgumentException("mark > position: ("
+ mark + " > " + pos + ")");
this.mark = mark;
}
} /**
* Returns this buffer's capacity.
*
* @return The capacity of this buffer
*/
//返回这个Buffer的容量
public final int capacity() {
return capacity;
} /**
* Returns this buffer's position.
*
* @return The position of this buffer
*/
//返回这个Buffer中当前的位置(当前操作数)
public final int position() {
return position;
} /**
* Returns this buffer's limit.
*
* @return The limit of this buffer
*/
//返回当前Buffer中可以被操作的元素的个数
public final int limit() {
return limit;
} /**
* Sets this buffer's mark at its position.
*
* @return This buffer
*/
//记录当前position的位置
public final Buffer mark() {
mark = position;
return this;
} public final Buffer reset() {
int m = mark;
if (m < 0)
throw new InvalidMarkException();
position = m;
return this;
} }

其中定义了四个Buffer属性,对应的描述如下

属性 描述
capacity 容量;用于描述这个Buffer大小,即创建的数组的长度,一旦声明不可以被改变
position 位置,表示当前缓冲区中正在操作的数据的位置,在切换读取时会将其置0
limit 界限、限制;表示当前缓冲区中可以操作的数据的大小,默认情况下为Buffer的大小,切换为读取模式后为数组中元素的个数(准确的说时切换之前position的值)
mark 标记;用于记录当前position的位置,后续操作过程中可以使用reset()方法将position还原至最后一次mark的位置

Buffer的创建与使用(ByteBuffer为例)

Buffer的创建

Java NIO中可以使用对应Buffer类的allocate()或者allocateDirect()静态方法创建。

//使用allocate()创建
ByteBuffer byteBuffer=ByteBuffer.allocate(1024); //使用allocateDirect()创建
ByteBuffer byteBuffer = ByteBuffer.allocateDirect(1024);

Buffer的本质是一个数组,创建时需要指定数组的大小

Buffer的使用

Buffer的使用一般分为四个步骤

  1. Buffer中写入数据
  2. Buffer切换为读取模式
  3. 读取Buffer
  4. Buffer清空,供后续写入使用

1. 写如数据

//使用put()方法向Buffer中写入数据
byteBuffer.put("bmilk".getBytes()); //使用Channel#read()向Buffer中写入数据
channel.read(byteBuffer);

2. 将Buffer切换为读取模式

可以通过调用flip()方法将Buffer从写模式切换到读模式。

byteBuffer.flip()

调用flip()方法会将position设回0,并将limit设置成之前position的值。

即,现在使用position标记读的位置,limit表示之前写进了多少个byte,也就是现在

能读取多少个byte等。

3. 读取Buffer

读取Buffer有两种方式:

  1. Buffer种读取数据到Channel
  2. 使用get()方法从Buffer种读取数据
//从Buffe中将数据写入通道
inChannel.write(byteBuffer) //使用get()方法从BUffer中读取数据
byte[] bytes=new byte[byteBuffer.limit()];
byteBuffer.get(bytes);

4. 将Buffer清空,供后续写入使用

使用clear()清空缓冲区,清空缓冲区只是使各个指针恢复初始位置,

更具体的说是position设置为0,limit设置为容量的初始大小。

并不会真实清空其中数据,但是可以通过后续的写覆盖之前的数据

byteBuffer.clear()

其他的一些方法

  1. 使用rewind()Buffer重复读取数据
//使用`rewind()`从`Buffer`重复读取数据
//Buffer.rewind()将position设回0,所以你可以重读Buffer中的所有数据。
//limit保持不变,仍然表示能从Buffer中读取多少个元素(byte、char等)。
Buffer rewind = byteBuffer.rewind();
  1. compact()方法

clear()会使使各个指针恢复初始位置,但是实际中可能存在部分数据还没有被使用,而后续需要使用。

又必须清理一部分Buffer的空间,compact()方法会将所有未读数据拷贝到Buffer的起始处,

然后将position指针设置到最后一个未读元素的后面,现在Buffer可以进行写数据,

但是不会覆盖前面的未读的数据。

  1. mark()方法与reset()方法

通过调用Buffer.mark()方法,可以标记Buffer中的当前的position。之后可以通过调用Buffer.reset()方法恢复到这个position。

//使用mark标记当前的position位置
byteBUffer.mark()
//使用reset方法使position指针返回这个位置
byteBuffer.reset()

4.equals()方法与compareTo()方法

当需要比较两个Buffer时可以使用equals()方法与compareTo()方法。

equals()方法判断两个方式是否相等,当满足下列条件时,表示两个Buffer相等

  • 有相同的类型(bytecharint等)
  • Buffer中剩余的bytechar等的个数相等。
  • \(\color{#FF3030}{`Buffer`中所有剩余的`byte`、`char`等都相同}\)

compareTo()方法比较两个两个Buffer的大小,仅比较剩余元素(bytechar等)

如果满足下列条件,则认为一个Buffer“小于”另一个Buffer

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

直接缓冲区与非直接缓冲区

  • 非直接缓冲区:通过allocate()方法分配缓冲区,将缓冲区建立在JVM内存中
  • 直接俄缓冲区:通过allocateDirect()方法分配直接缓冲区,将缓冲区建立在物理内存中,可以在某些情况下提高效率

非直接缓冲区

  • 非直接缓冲区数据流向图

直接缓冲区

  • 直接缓冲区数据流向图

直接缓冲区(物理内存映射文件):相比非直接缓冲区省略了copy的过程,所以说直接缓区可以一定程度上提高效率

弊端:

  • 开辟空间时资源消耗大
  • 不安全,java程序将数据写入物理内存映射文件中,之后数据将不受Java程序控制,

    什么时候写入硬盘无法控制(由操作系统控制),当垃圾回收机制释放引用后才能断开与之的连接

小结

  • 缓冲区要么是直接的,要么是非直接的如果为直接字节缓冲区,则java虚拟机会见最大努力直接在此缓冲区上执行本机I/O

    也就是说,每次调用基础操作系统的I/O之前或之后,虚拟机都回尽量避免将缓冲区的内容复制到中间缓冲区或者从中间缓冲区中复制内容。
  • 直接字节缓冲区可以通过调用此类的allocateDirect()工厂方法来创建,

    此方法返回的缓冲区进行分配和取消分配所需的程本通常高于非直接缓冲区,

    直接缓冲区的内容可以驻留在常规的垃圾回收堆之外,因此他们对应用程序内存需求造成的影响可能并不明显,

    所以建议直接缓冲区主要分配给易受基础系统的本机I/O操作影响的大型、持久得缓冲区。

    一般情况下,最好尽在直接缓冲区能在程序性能方面带来明显好处时分配他们。
  • 直接字节缓冲区还可以通过FileChannelmap()方法,将文件区域直接映射到内存中来创建,

    该方法返回MappedByteBufferJava的实现有助于JNI从本地及代码创建直接字节缓冲区,

    如果以上这些缓冲区中的某个缓冲区实例指的是不可访问的内存区域。

    则试图访问该区域不会更改缓冲区的内容,并且将会在访问期间或稍后的时间导致抛出不确定的异常
  • 字节缓冲区是直接缓冲区还是非直接缓冲区可以通过调用其isDirect()方法来确定,提供此方法是为了能够在性能关键型代码中执行显式缓冲区管理。

总结

本文简单介绍了Buffer的种类,并对常用方法进行乐简单的介绍

参考资料

Java NIO系列教程(三) Buffer

Java NIO之Buffer的使用的更多相关文章

  1. Java NIO 之 Buffer

    Java NIO 之 Buffer Java NIO (Non Blocking IO 或者 New IO)是一种非阻塞IO的实现.NIO通过Channel.Buffer.Selector几个组件的协 ...

  2. Java NIO之Buffer(缓冲区)

    ​ Java NIO中的缓存区(Buffer)用于和通道(Channel)进行交互.数据是从通道读入缓冲区,从缓冲区写入到通道中的. ​ 缓冲区本质上是一块可以写入数据,然后可以从中读取数据的内存.这 ...

  3. JAVA NIO简介-- Buffer、Channel、Charset 、直接缓冲区、分散和聚集、文件锁

    IO  是主存和外部设备 ( 硬盘.终端和网络等 ) 拷贝数据的过程. IO 是操作系统的底层功能实现,底层通过 I/O 指令进行完成. Java标准io回顾 在Java1.4之前的I/O系统中,提供 ...

  4. java nio之Buffer(一)

    Buffer是一个包装了基本数据元素数组的对象,它以及它的子类定义了一系列API用于处理数据缓存. 一.属性 Buffer有四个基本属性: 1.capacity  容量,buffer能够容纳的最大元素 ...

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

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

  6. Java NIO 之 Buffer(缓冲区)

    一 Buffer(缓冲区)介绍 Java NIO Buffers用于和NIO Channel交互. 我们从Channel中读取数据到buffers里,从Buffer把数据写入到Channels. Bu ...

  7. 【Java nio】buffer

    package com.slp.nio; import org.junit.Test; import java.nio.ByteBuffer; /** * Created by sanglp on 2 ...

  8. java nio之Buffer

    一.JAVA NIO 是在和channel交互的时候使用的.Channel将数据读入缓冲区,然后我们又从缓冲区访问数据.写数据时,首先将要发送的数据按顺序填入缓冲区.基本上,缓冲区只是一个列表,它的所 ...

  9. Java NIO:Buffer、Channel 和 Selector

    Buffer 一个 Buffer 本质上是内存中的一块,我们可以将数据写入这块内存,之后从这块内存获取数据. java.nio 定义了以下几个 Buffer 的实现,这个图读者应该也在不少地方见过了吧 ...

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

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

随机推荐

  1. Java实现第十届蓝桥杯JavaC组第十题(试题J)扫地机器人

    扫地机器人 时间限制: 1.0s 内存限制: 512.0MB 本题总分:25 分 [问题描述] 小明公司的办公区有一条长长的走廊,由 N 个方格区域组成,如下图所 示. 走廊内部署了 K 台扫地机器人 ...

  2. 第七届蓝桥杯JavaC组省赛真题

    解题代码部分来自网友,如果有不对的地方,欢迎各位大佬评论 题目1.有奖竞猜 题目描述 小明很喜欢猜谜语. 最近,他被邀请参加了X星球的猜谜活动. 每位选手开始的时候都被发给777个电子币. 规则是:猜 ...

  3. Java实现旅行商问题

    1 问题描述 何为旅行商问题?按照非专业的说法,这个问题要求找出一条n个给定的城市间的最短路径,使我们在回到触发的城市之前,对每个城市都只访问一次.这样该问题就可以表述为求一个图的最短哈密顿回路的问题 ...

  4. java矩形的关系

    在编写图形界面软件的时候,经常会遇到处理两个矩形的关系. 如图[1.jpg]所示,矩形的交集指的是:两个矩形重叠区的矩形,当然也可能不存在(参看[2.jpg]). 两个矩形的并集指的是:能包含这两个矩 ...

  5. Java实现第九届蓝桥杯字母阵列

    字母阵列 题目描述 仔细寻找,会发现:在下面的8x8的方阵中,隐藏着字母序列:"LANQIAO". SLANQIAO ZOEXCCGB MOAYWKHI BCCIPLJQ SLAN ...

  6. PAT 有理数四则运算

    本题要求编写程序,计算 2 个有理数的和.差.积.商. 输入格式: 输入在一行中按照 a1/b1 a2/b2 的格式给出两个分数形式的有理数,其中分子和分母全是整型范围内的整数,负号只可能出现在分子前 ...

  7. portapack发射GPS的信号实现GPS脱机模拟器

    要注意portapack必须要购买带高精度晶振的版本,另外固件要刷gridRF版本,用官方的或者havoc的都不行. 固件在这下载: 链接: https://pan.baidu.com/s/16flB ...

  8. 【Python源码剖析】对象模型概述

    Python 是一门 面向对象 语言,实现了一个完整的面向对象体系,简洁而优雅. 与其他面向对象编程语言相比, Python 有自己独特的一面. 这让很多开发人员在学习 Python 时,多少有些无所 ...

  9. 【asp.net core 系列】5 布局页和静态资源

    0. 前言 在之前的4篇的内容里,我们较为详细的介绍了路由以及控制器还有视图之间的关系.也就是说,系统如何从用户的HTTP请求解析到控制器里,然后在控制器里处理数据,并返回给视图,在视图中显示出来.这 ...

  10. CORS跨域漏洞学习

    简介 网站如果存CORS跨域漏洞就会有用户敏感数据被窃取的风险. 跨域资源共享(CORS)是一种浏览器机制,可实现对位于给定域外部的资源的受控访问.它扩展了同源策略(SOP)并增加了灵活性.但是,如果 ...