【发布时间】:2012-04-29 03:49:53
【问题描述】:
读取(用c++程序)fortran代码生成的二进制文件的问题被问了很多次,对fortran记录中的约定给出了令人满意的描述(例如http://local.wasp.uwa.edu.au/~pbourke/dataformats/fortran/)
但是,当我尝试实现 c++ 程序时,请记住 fortran 约定它仍然不起作用。在这里,我假设我们二进制文件“test.bin”包含 1 个整数,并由 fortran 例程以二进制格式写入。 以下是我尝试在 C++ 中阅读它的方法:
#include <iostream>
#include <fstream>
using namespace std;
int main () {
ifstream file;
file.open("test.bin", ios::in|ios::binary);
if (file.is_open())
{
int ival;
file >> ival >> ival; cout<< ival<<endl;
}
return 0;
}
这里的双 >>ival 构造首先读取 fortran 记录的标头(其中包含记录的大小(以字节为单位)),第二个 >>ival 应该提取值。文件中写入的整数是8,但是程序输出的是0,所以没有正确读取数据。
这是二进制文件的内容: ^D^@^@^@^@^@^@^@^H^@^@^@^D^@^@^@^@^@^@^@
所以我的问题 - 我做错了什么?
这是十六进制编辑器显示的内容:
0000000: 0400 0000 0000 0000 0800 0000 0400 0000 ................
0000010: 0000 0000 0a .....
知道这意味着什么吗?
【问题讨论】:
-
十六进制!将字节输出为字符没有任何帮助。
-
判断它是大端还是小端。英特尔是小端。
^@ = 0, ^D = 4, ^H = 8。在我看来像 little endian,记录大小是 4,值是 8。我不知道最后 8 个字节是什么。 -
>>运算符读取文本数据(即,它尝试解析整数的十进制表示)。使用read成员函数读取二进制数据。 -
嗯,第二个整数,假设32位little-endian,其实是0,接下来是8,然后是4。如果文件中写入的第一个整数实际上是8,可能是大小header 是一个 64 位整数 - 对于大型磁盘文件来说非常合理。