我正在测试二进制文件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位模式下重新编译并比较二进制文件程序生成)。