ByteBuffer缓冲区的三种创建方式与解读


字节缓冲区分为直接字节缓冲区与非直接字节缓冲区。

1、allocate()创建堆缓冲区

  ByteBuffer buffer1 = ByteBuffer.allocate(10);   

allocate()方法的作用:

  • 分配一个新的非直接字节缓冲区。
  • position为0,limit为capacity,mark不确定。
  • 它将具有一个底层实现数组,且数组偏移量为0。
底层api
public static ByteBuffer allocate(int capacity) {
    if (capacity < 0)
        throw new IllegalArgumentException();
    return new HeapByteBuffer(capacity, capacity);
}

class HeapByteBuffer
    extends ByteBuffer
{
HeapByteBuffer(int cap, int lim) {
    super(-1, 0, lim, cap, new byte[cap], 0);
}
}

2、allocateDirect()创建直接缓冲区

   ByteBuffer buffer2 = ByteBuffer.allocateDirect(10);

allocateDirect()的作用:

  • 分配新的直接字节缓冲区。
  • position为0,limit为capacity,mark不确定。
  • 在JVM层面没有实现数组。
底层api
public static ByteBuffer allocateDirect(int capacity) {
    return new DirectByteBuffer(capacity);
}
class DirectByteBuffer  extends MappedByteBuffer  implements DirectBuffer

3、wrap()创建堆缓冲区

  ByteBuffer buffer3 = ByteBuffer.wrap(new byte[]{1,2,3,4},0,3);

wrap(byte[]array) 方法的作用:

  • 将byte数组包装到缓冲区中。
  • 新的缓冲区将由给定的byte数组支持,也就是说,缓冲区修改将导致数组的修改,反之亦然。
  • 新缓冲区的capacity和limit将为array.length,其位置position将为0,其标记mark是不确定的。
  • 其底层实现数组将为给定数组,并且其arrayOffset将为0。

wrap(byte[]array,int offset,int length)方法的作用:

  • 新缓冲区的capacity将为array.length,其position为offset(0<offset<=array.length),limit为offset+length(0<length<=array.length-offset)
  • 其不具备subString()方法截取功能,它的参数offset只是设置缓冲区的position值,而length确定limit值。
  • (其他与上个方法wrap(byte[]array) 一致)
public static ByteBuffer wrap(byte[] array,
                                int offset, int length)
{
    try {
        return new HeapByteBuffer(array, offset, length);
    } catch (IllegalArgumentException x) {
        throw new IndexOutOfBoundsException();
    }
}

HeapByteBuffer(byte[] buf, int off, int len) {
    super(-1, off, off + len, buf.length, buf, 0);
}

4、直接缓冲区和非直接缓冲区的比较:

直接缓冲区:

  • 直接缓冲区在内部使用sun.misc.Unsafe类进行值的处理。Unsafe类的作用是JVM与操作系统进行直接通信。
  • 直接缓冲区操作的数据不在JVM堆中,而是在内核空间中,这样提高运行效率。
  • 通过allacateDirect()返回的缓冲区进行内存的分配和释放所需的时间成本通常要高于非直接缓冲区,但运行效率远比非直接高。

非直接缓冲区:

  • 在内部直接对数组进行操作,并且是在JVM的堆中进行数组处理。

直接缓冲区提高运行效率的原理是每次调用基于操作系统的I/O操作之前或之后,JVM都会尽量避免将缓冲区的内容复制到中间缓冲区,或者从中间缓冲区中获取内容,这样就节省了一个步骤。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

徐同学呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值