今日,写一个小系统时,因为结构体过长,欲用二进制一次性保存一个节点,结果在读取文件时却出现了一堆 屯,
这就很尴尬了。
左思右想,右思左想,百度完,去谷歌,谷歌完,又必应,必应问完贴吧问,贴吧问完CSDN问,
最后还是没有解决方案,这就很悲催了。
先放上部分源代码
#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <string.h> #include <stdlib.h> #include <conio.h> #include <windows.h> typedef struct student { int score; int num; char name[30]; int _class; char major[30]; char academy[30]; char sex; int height; int weight; int birth_year; int birth_month; int birth_day; char birthplace[50]; /*char home_place[50]; char guardian[30];//监护人 char tell;*/ student *next; }student; void InitList(student *&L) { L = (student *)malloc(sizeof(student)); L->next = NULL; } void Add(student *&L) { int ch; student *p, *q; q = L; p = (student *)malloc(sizeof(student)); //插入,直接在新增节点时排序 //学号,姓名,班级,专业,所在院系,性别,身高,体重,出生日期,籍贯 printf("请输入姓名\n"); scanf("%s", p->name); printf("请输入学号\n"); scanf("%d", &p->num); printf("请输入素拓分\n"); scanf("%d", &p->score); printf("%d\n", p->score); printf("请输入班级\n"); scanf("%d", &p->_class); printf("请输入专业\n"); scanf("%s", p->major); printf("请输入院系\n"); scanf("%s", p->academy); rewind(stdin); printf("请输入性别(W或M)\n"); scanf("%c", &p->sex); printf("请输入身高(cm)\n"); scanf("%d", &p->height); printf("请输入体重(kg)\n"); scanf("%d", &p->weight); printf("请输入出生日期(年)\n"); scanf("%d", &p->birth_year); printf("请输入出生日期(月)\n"); scanf("%d", &p->birth_month); printf("请输入出生日期(日)\n"); scanf("%d", &p->birth_day); printf("请输入籍贯\n"); scanf("%s", &p->birthplace); if (L->next == NULL) { p->next = L->next; L->next = p; } else { while (q->next != NULL && p->score < q->next->score) q = q->next; p->next = q->next; q->next = p; } printf("是否继续添加?(Y/N)\n"); ch = _getch(); if (ch == 'Y' || ch == 'y') Add(L); } void Save(student *L) { FILE *fp; student *p; if ((fp = fopen("信息.txt", "wb")) == NULL) { printf("保存失败!\n"); return; } for (p = L->next; p != NULL; p = p->next) fwrite(p,sizeof(student),1,fp);//fwrite()函数返回成功写入项的数量。正常情况下,该返回值就是 nmemb,但如果出现写入错误,返回值会比nmemb小。 fclose(fp); } void Load(student *&L) {//因为保存时,先保存大,再保存小,为了保持原有顺序,需要在读取时采用尾插法 student *q; q = L; FILE *fp; if ((fp = fopen("信息.txt", "rb+")) == NULL) { printf("读取失败!\n"); return; } CNodeR(q, fp); printf("读取成功!\n"); fclose(fp); } void CNodeR(student *&L, FILE *fp) { student *p = (student *)malloc(sizeof(student)); fread(p, sizeof(student), 1, fp);
p->next = L->next; L->next = p; if (!feof(fp)) CNodeR(L, fp); }
屯屯屯!!!输入法按 t 第一个都是屯了...
然后调试发现,保存过程没有什么问题,;读取时看起来也没有什么问题
可是!本应该读取完的数据还在继续读取!
于是想出一个快速解决方案
void Load(student *&L) {//因为保存时,先保存大,再保存小,为了保持原有顺序,需要在读取时采用尾插法 student *q; q = L; FILE *fp; if ((fp = fopen("信息.txt", "rb+")) == NULL) { printf("读取失败!\n"); return; } CNodeR(q, fp); printf("读取成功!\n"); fclose(fp); Sleep(1000); } void CNodeR(student *&L, FILE *fp) { student *p = (student *)malloc(sizeof(student)); fread(p, sizeof(student), 1, fp); if (p->score < -1000) { free(p); return;//这个判断是为了取消fread的副作用:多读一个节点 } p->next = L->next; L->next = p; if (!feof(fp)) CNodeR(L, fp); }
问题解决。
结论,多观察调试。