【发布时间】:2017-09-01 18:01:18
【问题描述】:
我目前正在开发可以读取二进制文件的程序。我知道有很多主题可以回答同一个问题,但我的问题非常单一,我还没有找到任何答案。
所以我知道二进制文件的结构:
- 4 字节:文件中的表数 然后是文件中的第一个表:
- 4 字节:表中的数据类型
- 4字节:表名长度
- L 字节:表名
- 4 字节:行数
- 4 字节:列数
- Nbr 行 * Nbr 列 * sizeof(type) 字节:数据
因此,使用读取指令,我设法获取了数据之前的所有内容。 例如,在我的第一个二进制文件中,我的第一个表是浮点数 3 行和 300 000 列。我设法获得了前 66 个,然后我在第 67 个获得了 endofFile 标志,并为我尝试读取的所有 899 932 个其他浮点数获得了坏字节标志。
这是我的标题代码的一些部分(效果很好)
uint32_t tableManager::getNbrTables()
{
uint32_t a;
file.read(reinterpret_cast<char *>(&a), sizeof(a));
return a;
}
uint32_t tableManager::getTypeData(int k)
{
uint32_t a;
file.read(reinterpret_cast<char *>(&a), sizeof(a));
return (a - 1);
}
这些为我提供了我需要的标题的正确值。然后我使用循环来获取数据值,代码如下:
vector<vector<float>> tmpL(nbrL[m]);
vector<float> tmpC(nbrC[m]);
switch (typeData[m])
{
case 0:
char x0;
for(int n = 0; n < nbrL[m]; n++)
{
for(int o = 0; o < nbrC[m]; o++)
{
file.read(reinterpret_cast<char *>(&x0), sizeof(x0));
tmpC.push_back(x0);
}
tmpL.push_back(tmpC);
}
dataT.push_back(tmpL);
break;
case 1:
float x1;
for(int n = 0; n < nbrL[m]; n++)
{
for(int o = 0; o < nbrC[m]; o++)
{
file.read(reinterpret_cast<char *>(&x1), sizeof(x1));
tmpC.push_back(x1);
}
tmpL.push_back(tmpC);
}
dataT.push_back(tmpL);
break;
}
在调用函数 m = 0 时,这意味着它是数据中两个表中的第一个。
但我不明白为什么数据读取开始有效,然后在几次读取后停止工作。 根据我使用的二进制文件,标题总是被正确读取,但读取的浮点数的数量会有所不同,即使至少读取了两个。
我尝试使用 seekg() 手动放置读取点,但效果完全一样。
如果您发现了什么或需要更多信息,感谢您的回答
【问题讨论】:
-
在 Windows 上,您需要使用
"b"标志打开文件以确保不会发生文本解释。 -
哇,非常感谢,我确定我做到了,我今天早些时候一定已经改变了......对不起,那么长的帖子!我是堆栈溢出的新手,如何接受答案?
-
应该有一个“接受”按钮。
-
我是堆栈溢出的新手 -- 好吧,我为您实际进行合法、经过深思熟虑的二进制文件读/写而表扬。这里有很多与此相关的问题涉及将 STL 容器和
std::strings保存在二进制文件中,并想知道为什么程序无法正确读取/写入。 -
值是大端还是小端?