有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

bytearray Java位运算符未按预期工作

下面的代码试图在字节数组中存储4个long。随机访问很重要,这就是为什么我不使用字节流。为什么下面的代码不起作用?你能想出一个更有效的方法吗

public static void storeLong(long value, byte[] buf, int offset) {
    buf[offset] = (byte) (value & 0xFFL);
    buf[offset+1] = (byte) ((value >>> 8) & 0xFFL);
    buf[offset+2] = (byte) ((value >>> 16) & 0xFFL);
    buf[offset+3] = (byte) ((value >>> 24) & 0xFFL);
    buf[offset+4] = (byte) ((value >>> 32) & 0xFFL);
    buf[offset+5] = (byte) ((value >>> 40) & 0xFFL);
    buf[offset+6] = (byte) ((value >>> 48) & 0xFFL);
    buf[offset+7] = (byte) ((value >>> 56) & 0xFFL);
}

public static long retrieveLong(byte[] buf, int offset) {
    return ((long)buf[offset]) 
        + (((long)buf[offset+1])<<8) 
        + (((long)buf[offset+2])<<16) 
        + (((long)buf[offset+3])<<24)
        + (((long)buf[offset+4])<<32) 
        + (((long)buf[offset+5])<<40) 
        + (((long)buf[offset+6])<<48) 
        + (((long)buf[offset+7])<<56);
}


public static void main(String[] args) {
    byte[] buf = new byte[32];
    storeLong(-1, buf, 0);
    storeLong(1, buf, 8);
    storeLong(Long.MAX_VALUE, buf, 16);
    storeLong(Long.MIN_VALUE, buf, 24);

    System.out.println(-1);
    System.out.println(1);
    System.out.println(Long.MAX_VALUE);
    System.out.println(Long.MIN_VALUE);

    System.out.println(retrieveLong(buf, 0));
    System.out.println(retrieveLong(buf, 8));
    System.out.println(retrieveLong(buf, 16));
    System.out.println(retrieveLong(buf, 24));

}

我从上面得到的结果如下。您可以看到前四个数字与下四个不匹配:

-1
1
9223372036854775807
-9223372036854775808

-72340172838076673
1
9151031864016699135
-9223372036854775808

共 (2) 个答案

  1. # 1 楼答案

    不要使用+并且byte已签名:

    public static long retrieveLong(byte[] buf, int offset) {
        return ((long)buf[offset]     & 255) 
            | (((long)buf[offset + 1] & 255) << 8) 
            | (((long)buf[offset + 2] & 255) << 16) 
            | (((long)buf[offset + 3] & 255) << 24)
            | (((long)buf[offset + 4] & 255) << 32) 
            | (((long)buf[offset + 5] & 255) << 40) 
            | (((long)buf[offset + 6] & 255) << 48) 
            | (((long)buf[offset + 7] & 255) << 56);
    }
    

    你必须and每个字节加255才能使其成为“无符号”。此外,还必须使用二进制or而不是add

  2. # 2 楼答案

    我做了一些测试,发现使用java。尼奥。LongBuffer的速度是我的代码的两倍

        ByteBuffer bb = ByteBuffer.allocate(4*8);
        LongBuffer lb = bb.asLongBuffer();
    
        lb.put(0, -1);
        lb.put(1, 1);
        lb.put(2, Long.MAX_VALUE);
        lb.put(3, Long.MIN_VALUE);
    
        System.out.println(lb.get(0));
        System.out.println(lb.get(1));
        System.out.println(lb.get(2));
        System.out.println(lb.get(3));
    

    然后我可以使用bb获得字节数组。数组()

    感谢Louis Wasserman和Rene Jeschke的努力