c语言链表fwrite二进制保存,读取时出现 屯 的问题

今日,写一个小系统时,因为结构体过长,欲用二进制一次性保存一个节点,结果在读取文件时却出现了一堆 屯,

这就很尴尬了。

左思右想,右思左想,百度完,去谷歌,谷歌完,又必应,必应问完贴吧问,贴吧问完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);
}

问题解决。

结论,多观察调试。

原文地址:https://www.cnblogs.com/luoshui/p/9402835.html