我正在测试二进制文件 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
的结果时,我得到:
我知道在二进制文件中,它不会以人类可读的形式显示它,但它应该看起来像这样吗?我不这么认为,因为年龄甚至不可见,并且 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/