有 Java 编程相关的问题?

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

python如何将C风格的二进制浮点数组读入Java?

我有一个二进制文件,其中包含8个字节的浮点(双字节)。当我使用以下代码在python中阅读时:

import array
d = array.array('d')
d.fromfile(open("foo", mode="rb"), 10)
print d

我得到的结果与在同一文件上运行的java代码不同:

DataInputStream is;

try {
    is = new DataInputStream(new FileInputStream(FILE_NAME));

    int n = 0;

    while(n < 10) {
        System.out.println(is.readDouble());
        n++;
    }

    is.close();

} catch (FileNotFoundException e) {
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}

我做错了什么

以下是示例输出:

爪哇:

-6.519670308091451E91
-6.723367689137016E91
0.0
-6.519664091503568E91
1.2029778888642203E-19
1.2029778888642203E-19
1.2028455399662426E-19
-1.1421078747242632E217
2.2734939098318505E236
-3.494281168153119E125

Python:

array('d', [-1.504936576164858e-36, -5.878658489696332e-39, 0.0, -5.878658478748688e-39, 5.878658495170291e-39, 5.878658495170291e-39, -5.878655692573363e-39, -5.87865851296011e-39, 4.79728723e-315, 1.546036714e-314])

下面是我用来生成数据的C程序:

#include <stdio.h>

double test_data[10] = {
    -1.504936576164858e-36, 
    -5.878658489696332e-39, 
    0.0, 
    -5.878658478748688e-39, 
    5.878658495170291e-39, 
    .878658495170291e-39, 
    -5.878655692573363e-39, 
    -5.87865851296011e-39, 
    4.79728723e-315, 
    1.546036714e-314
};

int main() {
    FILE * fp;
    fp = fopen("foo", "wb");
    if(fp != NULL) {
        fwrite(test_data, sizeof(double), 10, fp);
        fclose(fp);
    }
    return 0;
}

共 (2) 个答案

  1. # 1 楼答案

    Java将doubles读取为大端,而您的数据可能是小端。要在Java中读取一点endian double,可以使用:

    double d = Double.longBitsToDouble(Long.reverseBytes(is.readLong()));
    

    这将以long的形式从文件中读取八个字节,将各个字节的顺序从一个字节交换到另一个字节,然后将值转换为double

  2. # 2 楼答案

    Java的DataInputStream总是将数据视为big-endian。使用Python的^{} module和示例数据可以看到这一点:

    >>> s = struct.pack("<d", -1.504936576164858e-36)
    >>> s
    '\xd3\x00\x00\xb9\xd3\x00\x80\xb8'
    >>> struct.unpack("<d", s)
    (-1.504936576164858e-36,)
    >>> struct.unpack(">d", s)
    (-6.519670308091451e+91,)
    

    您需要回答的问题是,您正在查看的数据是存储为小端还是大端,并正确解释它