当前位置: 首页 > article >正文

java.nio.ByteBuffer的 capacity, limit, position, mark

java.nio.ByteBuffer的 capacity, limit, position, mark

  1. Capacity(容量)
    定义:缓冲区的总容量,即缓冲区中可以容纳的元素的数量。这个容量在缓冲区创建时被设定,并且之后不能被改变。
    用途:它定义了缓冲区可以容纳多少数据。一旦缓冲区被分配,其容量就固定了。
  2. Limit(限制)
    定义:第一个不应该被读或写的元素的索引,换句话说,是缓冲区的当前终点。
    用途:在写模式下,limit 等于 capacity(表示你不能再往缓冲区中写数据了)。在读模式下,limit 会被设置为某个小于 capacity 的值,表示有多少数据是可读的。limit 可以被改变以反映缓冲区中实际数据的数量。
  3. Position(位置)
    定义:下一个要被读或写的元素的索引。
    用途:它指定了下一个操作(读或写)发生的位置。当从缓冲区读取数据时,position 会自动增加,指向下一个可读的数据。当向缓冲区写入数据时,position 同样会自动增加,指向下一个要写入的位置。
  4. Mark(标记)
    定义:一个备忘位置,通过调用 mark() 方法可以设置。
    用途:在某些操作之后,你可能想回到某个特定的位置,这时就可以使用 mark() 方法来标记当前的位置。之后,无论 position 走到哪里,你都可以通过调用 reset() 方法来回到 mark 标记的位置。注意,mark 是可选的,并且只有在 position 被改变之前,它才是有效的。

初始值: limitcapacity初始值相同, position初始是0, mark是-1

从测试代码可看出: limitcapacity初始值相同, position初始是0, mark是-1

测试代码

import java.nio.ByteBuffer;

public class TestByteBuffer2409172015 {
	public static boolean pln=true;
	public static void pln(Object...oAr) {
		if(pln)for(Object o:oAr)System.out.println(o);
	}

	
	public static void main(String...arguments) {
		ByteBuffer bbf = ByteBuffer.allocate(102400);
		
		pln(bbf);
		pln("bbf.capacity()"+bbf.capacity());
		pln("bbf.limit()"+bbf.limit());
		pln("bbf.position()"+bbf.position());
		pln("bbf.mark()"+bbf.mark());
		
	}
}

结果:

java.nio.HeapByteBuffer[pos=0 lim=102400 cap=102400]
bbf.capacity()102400
bbf.limit()102400
bbf.position()0
bbf.mark()java.nio.HeapByteBuffer[pos=0 lim=102400 cap=102400]

可看出:
实现类是: java.nio.HeapByteBuffer
limitcapacity初始值相同
position初始是0
mark()方法不是返回mark属性的值, 而是设置mark=position , 与reset()方法配合使用, mark初始值是-1,在不调用mark()方法就调用reset()方法时,会报错


从源码看出: limitcapacity初始值相同, position初始是0, mark是-1

这四个属性定义在ByteBuffer的父类 java.nio.Buffer中

    private int mark = -1;
    private int position = 0;
    private int limit;
    private final int capacity;

mark初始值是-1 , position初始值是0

ByteBuffer的两个静态工厂方法

    public static ByteBuffer allocateDirect(int capacity) {
        return new DirectByteBuffer(capacity);
    }

    public static ByteBuffer allocate(int capacity) {
        if (capacity < 0)
            throw createCapacityException(capacity);
        return new HeapByteBuffer(capacity, capacity, null);
    }
  • allocate(int capacity) 方法 new一个继承自ByteBufferHeapByteBuffer类, capacity属性与limit属性值都来自capacity参数值
  • allocateDirect(int capacity) new一个DirectByteBuffer类 extends MappedByteBuffer extends ByteBuffer
        DirectByteBuffer(int cap) {                   // package-private
    
        super(-1, 0, cap, cap, null);
        boolean pa = VM.isDirectMemoryPageAligned();
        int ps = Bits.pageSize();
        long size = Math.max(1L, (long)cap + (pa ? ps : 0));
        Bits.reserveMemory(size, cap);
    
        long base = 0;
        try {
            base = UNSAFE.allocateMemory(size);
        } catch (OutOfMemoryError x) {
            Bits.unreserveMemory(size, cap);
            throw x;
        }
        UNSAFE.setMemory(base, size, (byte) 0);
        if (pa && (base % ps != 0)) {
            // Round up to page boundary
            address = base + ps - (base & (ps - 1));
        } else {
            address = base;
        }
        try {
            cleaner = Cleaner.create(this, new Deallocator(base, size, cap));
        } catch (Throwable t) {
            // Prevent leak if the Deallocator or Cleaner fail for any reason
            UNSAFE.freeMemory(base);
            Bits.unreserveMemory(size, cap);
            throw t;
        }
        att = null;
    
    super(-1, 0, cap, cap, null); 这句看出 capacitylimit初始值相同

clear()方法也可看出: limitcapacity初始值相同, position初始是0, mark是-1

    public Buffer clear() {
        position = 0;
        limit = capacity;
        mark = -1;
        return this;

http://www.kler.cn/news/314443.html

相关文章:

  • 如何打造高效的远程开发团队:最佳实践与挑战
  • 大话C++:第11篇 类的定义与封装
  • Redis——redispluspls库通用命令以及String类型相关接口使用
  • 每日一题--打印闰年
  • 如何使用 Python 连接 MySQL 数据库?什么是 ORM(对象关系映射),如何使用
  • fasterRCNN模型实现飞机类目标检测
  • 果蔬识别系统架构+流程图
  • Hadoop的安装
  • JVM 调优篇7 调优案例2-元空间的优化解决
  • 使用Diskgenius系统迁移
  • 分页插件、代码生成器
  • C#中DataGridView 的 CellPainting 事件的e.Handled = true
  • 银河麒麟V10系统崩溃后的处理
  • 富文本编辑器wangEdittor使用入门
  • string类的模拟实现以及oj题
  • Linux·权限与工具-git与gdb
  • Puppet 部署应用(Puppet deployment application)
  • 《他们的奇妙时光》圆满收官,葛秋谷新型霸总获好评
  • 初始Vitis——ZYNQ学习笔记1
  • 探索微软Copilot Agents:如何通过Wave 2 AI彻底改变工作方式
  • 伊犁linux 创建yum 源过程
  • Java面向对象编程
  • Ubuntu设置笔记本电脑合盖时不挂起
  • el-select组件:选择某个选项触发查询
  • 基于R语言的统计分析基础:使用键盘输入数据
  • charles抓包flutter
  • 数据结构之线性表——LeetCode:328. 奇偶链表,86. 分隔链表,24. 两两交换链表中的节点
  • 基于React+JsonServer+Antddesign的读书笔记管理系统
  • 4.使用 VSCode 过程中的英语积累 - View 菜单(每一次重点积累 5 个单词)
  • 微软AI核电计划