struct book
{
unsigned short size_of_content;
unsigned short price;
unsigned char *content;
};
假设我的文件包含多个 book
,每个都有不同的 size_of_content
、price
和 content
。我怎样才能一次读一本书
并确定它是哪本书(例如查看价格)?
size_t nread2;
struct book *buff = malloc(sizeof(struct book));
while( (nread2 = fread(buff, sizeof(struct book), 1, infp)) > 0 )
{
printf("read a struct once \n");
}
这就是我目前所拥有的。每当我阅读一个结构时,我都会尝试打印。但是,当我尝试输入包含 5 个结构的文件时,它将打印 15 次...
谢谢。
最佳答案
让我们看看您的 struct
并想一想它有多大。
struct book {
unsigned short size_of_content;
unsigned short price;
unsigned char *content;
};
第一项是 unsigned short
,没问题,sizeof(unsigned short)
可能为 2,即 2 个字节。下一个也是如此。
但是第三个。那是指向 unsigned char
的指针...您的磁盘记录不太可能是保存的指针。您有一个字段 size_of_content
...我猜磁盘记录包含 size_of_content
,然后是 price
,然后是实际内容。
我不会为您编写完整的代码,但在伪代码中它是这样的:
fread(&size_of_content, sizeof(size_of_content), 1, infp)
sanity-check the value of size_of_content and handle any error
fread(&price, sizeof(price), 1, infp)
sanity-check teh value of price and handle any error
buff->content = malloc(size_of_content)
check for error on malloc and handle any error
fread(buff->content, size_of_content, 1, infp)
如果您对内容的大小没有严格的规范,只需假设它不会超过 10 亿或类似的数量,并确保该数字至少不会超过 10 亿那么大!始终检查错误。
由于 struct
中只有两个字段,因此很容易对每个字段进行 fread()
。如果你有一个更复杂的结构,使用结构可能是值得的:
struct book_header {
unsigned short size_of_content;
unsigned short price;
};
struct book {
struct book_header header;
unsigned char *content;
}
然后你可以使用 fread()
和 sizeof(book_header)
一次性读取整个标题。在处理波形音频文件等二进制文件时,我编写了很多这样的代码。
您可能不需要担心这一点,但如果文件是在“大端”计算机上写入并在“小端”计算机上读取,反之亦然,这将是一个问题。
http://en.wikipedia.org/wiki/Endianness
如果您确实遇到了这个问题,那么解决方案就是标准化。选择其中一个(小端或大端)并使用 C 库函数来确保使用该字节序写入和读取数字。比如写的时候是htonl()
库函数,读的时候是ntohl()
。
http://linux.die.net/man/3/htonl
但正如我所说,您可能不需要为此担心。
关于c - 如何 fread() 结构?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21721808/