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

时间:2014-12-10 23:08:58

标签: 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表明它看起来像是垃圾。

1 个答案:

答案 0 :(得分:2)

此代码段有两个不同的问题:首先,正如BLUEPIXY已经注释过,您需要关闭并重新打开或回放文件,然后再次读取它。否则,您的fread()会失败,并且由于未在程序中检查错误,因此在您尝试打印cpy_me内未完全未初始化的字符串之前,您将不会注意到。

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

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