c - 如何 fread() 结构?

标签 c file file-io struct io

struct book
{
    unsigned short  size_of_content;
    unsigned short  price;
    unsigned char  *content;
};

假设我的文件包含多个 book,每个都有不同的 size_of_contentpricecontent。我怎样才能一次读一本并确定它是哪本书(例如查看价格)?

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/

相关文章:

c - 给结构体数组赋值的问题!

javascript - Node读取ts模块/读取代码文件

python - select.select() 与常规文件

Java:将文本文件输出到控制台

C printf 变量的地址

c++ - 指向另一个结构的结构指针成员会导致段错误 (C/C++)

c - 如何使用 libuv 编译最小程序?

java - 即使单元格具有值,cellIterator.hasNext() 也会返回 false

java - 创建并写入文件

java - 自动注释 Java 类的程序/脚本