c - fwrite 打印垃圾,fread 读取垃圾结构

标签 c struct fwrite fread

我正在测试二进制文件 I/O。因此,为了练习,我编写了一个小程序:

main.c:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "person.h"

int main()
{
    struct person me;
    struct person cpy_me;
    FILE *dataWrite, *dataRead;

    strcpy(me.fname, "john");
    strcpy(me.lname, "smith");
    me.age = 12;

    printf("%s %s %d\n", me.fname, me.lname, me.age);

    dataWrite = fopen("people.bin", "wb");
    if (fwrite(&me, sizeof( struct person), 1, dataWrite) != 1)
        fprintf(stderr, "Error!\n");
    printf("Wrote to the file\n");

    dataRead = fopen("people.bin", "rb");

    fread(&cpy_me, sizeof(struct person), 1, dataRead);
    printf("%s->Fname\n", cpy_me.fname);
    printf("%s->Lname\n", cpy_me.lname);
    printf("%d->Age\n", cpy_me.age);

    fclose(dataWrite);
    fclose(dataRead);

    return 0;
}

person.h:

#ifndef PERSON_H_INCLUDED
#define PERSON_H_INCLUDED

#define   MAXFIRST    10
#define   MAXLAST     20

struct person
{
    char lname[MAXLAST], fname[MAXFIRST];
    int age;
};
#endif // PERSON_H_INCLUDED

这是文件 people.bin 中出现的内容:

smith ÿÿb¸tÄ[½tà@ john @ à@                                       

当我读取显示fread的结果时,我得到:

enter image description here

我知道在二进制文件中,它不会以人类可读的形式显示它,但它应该看起来像这样吗?我不这么认为,因为年龄甚至不可见,并且 fread 显示它读取垃圾。

最佳答案

此代码片段有两个独立的问题:首先,正如 BLUEPIXY 已经评论的那样,您需要关闭并重新打开或倒带文件,然后才能再次读入。否则,您的 fread() 会失败,并且由于不会检查程序中的错误,因此您不会注意到,直到您尝试打印 cpy_me 中的字符串,其中包含仍未初始化。

这涉及到第二个问题:确保将 struct me 初始化为零,例如通过执行 struct person me = {"", "", 0}; 这应该可以处理您在二进制文件中看到的垃圾。 C 中的结构不会自动初始化,除非它们是全局变量或者您明确告诉编译器这样做。

顺便说一句:请注意,像这样执行二进制 I/O 是不可移植的,因为结构在不同的体系结构上可能具有不同的内存布局(要看到这一点,请尝试在 32 位和 64 位模式下重新编译并比较您的程序生成的二进制文件)。

关于c - fwrite 打印垃圾,fread 读取垃圾结构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27412548/

相关文章:

c - struct 中的 struct timespec

c - fwrite后目标文件为空c语言

c - 如何使用fwrite写入数据0xf0f0f0f0

c - 指向结构的嵌套指针包含指向包含结构的指针?

c - 错误: invalid type argument of '->' (have 'Personaggio' )

c - 通过函数传递结构

c - 二进制文件读取 - 写入不起作用

c - 从函数返回浮点值... (C)

c - 从任何角度来看,++i 和 i+=1 之间有什么区别

c++ - 当两者都是 32 位宽时,在 C(或 C++)中使用 `unsigned long` 和 `unsigned int` 是否存在可观察到的差异?