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

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

通过视图修改 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: 0→0 1→0 2→0 3→0 4→0 5→0 6→0 7→97 
charBuffer: 0→  1→  2→  3→a
shortBuffer: 0→0 1→0 2→0 3→24832 
intBuffer: 0→0 1→1627389952 
floatBuffer: 0→0.0 1→1.4757395E20 
longBuffer: 0→6989586621679009792 
doubleBuffer: 0→1.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

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

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

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/zaji/5659968.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-16
下一篇 2022-12-16

发表评论

登录后才能评论

评论列表(0条)

保存