zoukankan      html  css  js  c++  java
  • 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);
    }

    问题解决。

    结论,多观察调试。

  • 相关阅读:
    Git------解决右键不显示Git Bash Here问题
    AngularJS------Error: Cannot find module '@angular-devkit/core'
    AngularJS------命令行
    AngularJS------报错"The selector "app-user-item" did not match any elements"
    AngularJS------使用VSCode创建的Angular项目部署IIS
    JQuery------各种版本下载
    docker 卸载
    oauth2.0授权协议
    web service 简介
    Python 优雅获取本机 IP 方法
  • 原文地址:https://www.cnblogs.com/luoshui/p/9402835.html
Copyright © 2011-2022 走看看