通过视图修改 ByteBuffer --《JAVA编程思想》80

ByteBuffer 只能操作字节,无法产生任何基本类型,但 ByteBuffer 提供 asCharBuffer()、asShortBuffer()、asIntBuffer() 返回对应基本类型的视图,我们对视图的任何修改对会映射成为对 ByteBuffer 数据中的修改。

视图允许我们单个或批量的往 ByteBuffer 中写入数据。

  • 示例:
    asIntBuffer() 获取 int 类型的视图,通过 get() 方法读取数据,put() 方法写入数据。
public class IntBufferDemo {

    public static void main(String[] args) {
        //创建指定大小的ByteBuffer
        ByteBuffer buffer = ByteBuffer.allocate(1024);
        //获取int类型的视图
        IntBuffer intBuffer = buffer.asIntBuffer();
        //批量写入数据
        intBuffer.put(new int[]{11, 12, 13, 14, 15, 16, 17, 18, 19});
        //获取指定位置的数据
        System.out.println("index 3 = " + intBuffer.get(3));
        //覆盖指定位置的数据
        intBuffer.put(2, 33);
        //将开始读取的指针指向0,读取结束的指针指向当前写入位置
        intBuffer.flip();
        //遍历数据
        while (intBuffer.hasRemaining()) {
            System.out.println(intBuffer.get());
        }
    }

}
index 3 = 14
11
12
33
14
15
16
17
18
19

通过在 ByteBuffer 中建立不同基本类型的视图,读取到的数据会有什么不同嘛?让我们一起来看看下面这个例子。

public class ViewBuffers {

    public static void main(String[] args) {
        ByteBuffer byteBuffer = ByteBuffer.wrap(new byte[]{0, 0, 0, 0, 0, 0, 0, 'a'});
        //将读取的指针指向头部
        byteBuffer.rewind();
        //直接根据字节读取
        System.out.print("byteBuffer: ");
        while (byteBuffer.hasRemaining()) {
            System.out.print(byteBuffer.position() + "→" + byteBuffer.get() + " ");
        }
        System.out.println();
        byteBuffer.rewind();
        //获取char类型的视图
        CharBuffer charBuffer = byteBuffer.asCharBuffer();
        System.out.print("charBuffer: ");
        while (charBuffer.hasRemaining()) {
            System.out.print(charBuffer.position() + "→" + charBuffer.get() + " ");
        }
        System.out.println();
        byteBuffer.rewind();
        //获取short类型的视图
        ShortBuffer shortBuffer = byteBuffer.asShortBuffer();
        System.out.print("shortBuffer: ");
        while (shortBuffer.hasRemaining()) {
            System.out.print(shortBuffer.position() + "→" + shortBuffer.get() + " ");
        }
        System.out.println();
        byteBuffer.rewind();
        //获取int类型的视图
        IntBuffer intBuffer = byteBuffer.asIntBuffer();
        System.out.print("intBuffer: ");
        while (intBuffer.hasRemaining()) {
            System.out.print(intBuffer.position() + "→" + intBuffer.get() + " ");
        }
        System.out.println();
        byteBuffer.rewind();
        //获取float类型的视图
        FloatBuffer floatBuffer = byteBuffer.asFloatBuffer();
        System.out.print("floatBuffer: ");
        while (floatBuffer.hasRemaining()) {
            System.out.print(floatBuffer.position() + "→" + floatBuffer.get() + " ");
        }
        System.out.println();
        byteBuffer.rewind();
        //获取long类型的视图
        LongBuffer longBuffer = byteBuffer.asLongBuffer();
        System.out.print("longBuffer: ");
        while (longBuffer.hasRemaining()) {
            System.out.print(longBuffer.position() + "→" + longBuffer.get() + " ");
        }
        System.out.println();
        byteBuffer.rewind();
        //获取double类型的视图
        DoubleBuffer doubleBuffer = byteBuffer.asDoubleBuffer();
        System.out.print("doubleBuffer: ");
        while (doubleBuffer.hasRemaining()) {
            System.out.print(doubleBuffer.position() + "→" + doubleBuffer.get() + " ");
        }
    }

}
byteBuffer: 00 10 20 30 40 50 60 797 
charBuffer: 0123→a
shortBuffer: 00 10 20 324832 
intBuffer: 00 11627389952 
floatBuffer: 00.0 11.4757395E20 
longBuffer: 06989586621679009792 
doubleBuffer: 01.757388200993436E159 

ByteBuffer 由一个8字节的数组构成,通过不同的基本类型的视图读取到的数据和个数存在差异,这是什么原因造成的呢?

上述例子中 ByteBuffer 通过不同类型的视图读取的数据结构如下:

在这里插入图片描述

因为不同基本类型占用的长度不同(byte类型1字节、char 和 short 类型2字节、int 和 float 类型占4个字节、long 和 double 类型占8个字节),故通过视图读取时,返回的个数和值均不一致。

不同的机器可能会使用不同的字节排序方法来存储数据。主要分为两种形式:高位优先(big endian)将重要的字节存放在地址最低的存储器单元;低位优先(little endian)则是将重要的字节存放在地址最高的存储器单元。

ByteBuffer 默认是以高位优先的顺序存放数据,在网络传输过程中,通常也是使用高位优先,我们可以通过 ByteBuffer 的 order() 方法配合常量 ByteOrder.BIG_ENDIAN 和 ByteOrder.LITTLE_ENDIAN 来改变字节的存放次序。

比如,short 类型占2个字节,依次为:00000000、01100001,使用的是高位优先的存放次序,读取出来的值为 97;
在这里插入图片描述
若改为低位优先,则2个字节的位置需要调换,成为:01100001、00000000,读取出来的值为 24832;
在这里插入图片描述

  • 代码示例:
public class ByteBufferSort {

    public static void main(String[] args) {
        //创建容量2个字节的ByteBuffer
        ByteBuffer byteBuffer = ByteBuffer.wrap(new byte[2]);
        byteBuffer.putShort((short) 97);
        byteBuffer.flip();
        //查看默认存放次序
        System.out.println(byteBuffer.getShort());
        byteBuffer.rewind();
        //设置为低位优先
        byteBuffer.order(ByteOrder.LITTLE_ENDIAN);
        System.out.println(byteBuffer.getShort());
        byteBuffer.rewind();
        //设置为高位优先
        byteBuffer.order(ByteOrder.BIG_ENDIAN);
        System.out.println(byteBuffer.getShort());
    }

}
97
24832
97

本次分享至此结束,希望本文对你有所帮助,若能点亮下方的点赞按钮,在下感激不尽,谢谢您的【精神支持】。

若有任何疑问,也欢迎与我交流,若存在不足之处,也欢迎各位指正!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

BaymaxCS

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

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

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

打赏作者

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

抵扣说明:

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

余额充值