使用c语言单向链表实现简单学生成绩管理系统(3)

使用c语言单向链表实现简单学生成绩管理系统

现在这个简单系统不能保存数数据,不能退出,本文使用lo操作将链表中维护的数据保存到文件中,即退出保存,重新运行再从文件中提取数据。


#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>


//个人信息结构体,各科成绩可自定义添加
typedef struct grade_msg
{
        char name[16];
        char sex;
        int chinese;
        int math;
        int english;

}grade_msg_t;

//链表结构体
typedef struct student_msg
{
        grade_msg_t msg;
        struct student_msg * next;
}student_msg_t;

//定义操作类型枚举
enum operate_type
{
        INSERT_TYPE = 1,
        DELETE_TYPE,
        PRINTF_TYPE,
        MODIFY_TYPE,
        QUIT_TYPE
};

//创建链表头
student_msg_t * create_slist_head()
{
        student_msg_t * head;
        //申请空间
        head = (student_msg_t *)malloc(sizeof(student_msg_t));
        if(head == NULL)
        {
                printf("create slist head fail\n");
                return NULL;
        }
        //初始化
        head->next = NULL;

        return head;
}

//判断链表中有无数据
//返回1为空,0为非空
int isnull(student_msg_t *  head)
{
        if(head->next == NULL)
                return 1;
        else
                return 0;
}

//链表头添加数据
int insert_data(student_msg_t * head,grade_msg_t * msg)
{
        printf("enter insert msg\n");

        student_msg_t * new_node;

        new_node = (student_msg_t * )malloc(sizeof(student_msg_t));
        if(new_node == NULL)
        {
                printf("insert,create space fail\n");
                return -1;
        }

        strcpy(new_node->msg.name,msg->name);
        new_node->msg.sex = msg->sex;
        new_node->msg.chinese = msg->chinese;
        new_node->msg.math = msg->math;
        new_node->msg.english = msg->english;

        new_node->next = head->next;

        head->next = new_node;
        return 0;
}

//打印所有信息
int printf_list(student_msg_t *  head)
{
        if(isnull(head))
        {
                printf("no student msg\n");
                return -1;
        }
        while(head->next != NULL)
        {
                head = head->next;
                printf("name:%-8s,sex:%c,chinese:%d,math:%d,english:%d\n",head->msg.name,head->msg.sex,head->msg.chinese,head->msg.math,head->msg.english);
        }

        return 0;

}

//保存数据
int save_data(student_msg_t * head,char * buf)
{
        char str[64];
        if(isnull(head))
        {
                printf("no student msg\n");
                return -1;
        }

        while(head->next != NULL)
        {
                head = head->next;
                sprintf(str,"name:%s ,sex:%c,chinese:%d,math:%d,english:%d\n",head->msg.name,head->msg.sex,head->msg.chinese,head->msg.math,head->msg.english);
                strcat(buf,str);
        }

        printf("buf:%s\n",buf);

        return 0;
}

//链表中删除数据--根据姓名
int delete_data(student_msg_t * head,char * name)
{
        if(isnull(head))
        {
                printf("no student msg\n");
                return -1;
        }

        int flag = 0;
        student_msg_t * temp_node;
        temp_node = (student_msg_t*)malloc(sizeof(student_msg_t));
        if(temp_node == NULL)
        {
                printf("create temp node fail\n");
                return -1;
        }

        while(head->next != NULL)
        {
                if(0 == strncmp(head->next->msg.name,name,strlen(name)))
                {
                        //删除此节点
                        temp_node = head->next;

                        head->next = temp_node->next;

                        free(temp_node);

                        temp_node = NULL;

                        flag = 1; //判断找到需要删除的信息

                        break;

                }
                else
                        head = head->next;
        }

        if(flag == 0)
                printf("the list do not have the name\n");
        else
                printf("delete the name msg is success\n");

        return 0;
}



//链表中修改数据--根据姓名
int modify_data(student_msg_t * head,grade_msg_t * grade_msg)
{
        if(isnull(head))
        {
                printf("no student msg\n");
                return -1;
        }

        int flag = 0;

        while(head->next != NULL)
        {
                if(0 == strncmp(head->next->msg.name,grade_msg->name,strlen(grade_msg->name)))
                {
                        //修改此节点
                        head->next->msg.chinese = grade_msg->chinese;
                        head->next->msg.math = grade_msg->math;
                        head->next->msg.english = grade_msg->english;

                        flag = 1; //判断找到需要删除的信息

                        break;

                }
                else
                {
                        head = head->next;
                }
        }

        if(flag == 0)
                printf("the list do not have the name\n");
        else
                printf("modify msg success\n");

        return 0;
}

int main(int argc, const char *argv[])
{
        student_msg_t * msg_head;
        char msg_str[64];
        char name_str[16];
        int flag;
        grade_msg_t insert_msg;
        int fd;
        char buf_str[512];
        int ret = 0;

        //从文件中读字符串后解析使用
        char * in_str = NULL;
        char * residue_str = NULL;
        char * temp_str = NULL;

        //创建链表头
        msg_head = create_slist_head();
        if(msg_head == NULL)
        {
                printf("create student msg head fail\n ");
                return -1;
        }

        //创建一个文件描述符
        fd = open("./student_msg_file",O_RDWR|O_CREAT);
        if(fd < 0)
        {
                printf("open the file fail\n");
                return -1;
        }

		//从文件中读取数据,添加到链表
        memset(buf_str,0,512);
        while(0 < (ret = read(fd,buf_str,sizeof(buf_str))))
        {
                printf("buf_str:%s\n",buf_str);
                in_str = strstr(buf_str,"name:");

                while(NULL != (in_str = strtok_r(in_str,"\n",&residue_str)))
                {
                        printf("in str:%s\n",in_str);//in str:name:xiaoli  ,sex:b,chinese:85,math:75,english:96

                        sscanf(in_str,"name:%s ,sex:%c,chinese:%d,math:%d,english:%d",insert_msg.name,&insert_msg.sex,&insert_msg.chinese,&insert_msg.math,&insert_msg.english);

                        printf("n:%s,s:%c,c:%d,m:%d,e:%d\n",insert_msg.name,insert_msg.sex,insert_msg.chinese,insert_msg.math,insert_msg.english);

                        insert_data(msg_head,&insert_msg);

                        in_str = NULL;
                }
        }

        while(1)
        {
                printf("input operate flag \n");
                printf("1 INSERT_TYPE\n");
                printf("2 DELETE_TYPE\n");
                printf("3 PRINTF_TYPE\n");
                printf("4 MODIFY_TYPE\n");
                printf("5 QUIT_TYPE\n");

                scanf("%d",&flag);
                getchar();

                switch(flag)
                {
                        case INSERT_TYPE:
                        {
                                //姓名 字符串 性别 字符类型 语数英 整型
                                printf("please input 姓名[enter] 性别[enter] 语文[enter] 数学[enter] 英语[enter]\n");
                                //简化信息输入
                                printf("------------input start------------\n");
                                scanf("%s",insert_msg.name);
                                getchar();

                                scanf("%c",&insert_msg.sex);
                                getchar();

                                scanf("%d",&insert_msg.chinese);
                                getchar();

                                scanf("%d",&insert_msg.math);
                                getchar();

                                scanf("%d",&insert_msg.english);
                                getchar();
                                printf("------------input end------------\n");

                                //插入数据
                                insert_data(msg_head,&insert_msg);
                        }
                        break;
                        case PRINTF_TYPE:
                        {
                                printf("==========printf start==========\n");
                                if(msg_head -> next != NULL)
                                        printf_list(msg_head);
                                else
                                        printf("do not have student msg \n");
                                printf("==========printf end=========\n");
                        }
                        break;
                        case DELETE_TYPE:
                        {
                                printf("<delete>please enter delete name\n");
                                memset(name_str,0,sizeof(name_str));
                                scanf("%s",name_str);
                                getchar();

                                delete_data(msg_head,name_str);
                        }
                        break;
                        case MODIFY_TYPE:
                        {
                                printf("<modify>please input modify name\n");
                                memset(name_str,0,sizeof(insert_msg.name));
                                scanf("%s",insert_msg.name);
                                getchar();

                                scanf("%d",&insert_msg.chinese);
                                getchar();

                                scanf("%d",&insert_msg.math);
                                getchar();

                                scanf("%d",&insert_msg.english);
                                getchar();

                                modify_data(msg_head,&insert_msg);
                        }
                        break;
                        case QUIT_TYPE:
                        {
                                goto out;
                        }
                        break;
                        default:
                                printf("no have the operate type\n");
                }
        }
out:
        //将链表数据缓存到buf_str中
        memset(buf_str,0,512);
        save_data(msg_head,buf_str);

        write(fd,buf_str,sizeof(buf_str));

        close(fd);
        return 0;
}

直接编译运行,亲测ok,注意代码中定义了一个buf,为512个字节,所以文件最多保存512个字节,这个大小可修改。

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
设计课题一:班级成绩管理系统 一、 问题描述: 对一个有N个学生的班级,每个学生有M门课程。该系统实现对班级成绩的录入、显示、修改、排序、保存等操作的管理。 二、功能要求: 1、本系统采用一个结构体数组,每个数据的结构应当包括:学号、姓名、M门课程名称。 2、本系统显示这样的菜单: 请选择系统功能项: a、 成绩录入 b、 成绩显示 c、 成绩保存 d、 成绩排序 e、 成绩修改(要求先输入密码) f、 成绩统计 (1) 显示每门课程成绩最高的学生的基本信息 (2) 显示每门课程的平均成绩 (3) 显示超过某门课程平均成绩学生人数 g、 退出系统 3、执行一个具体的功能之后,程序将重新显示菜单。 4、将学生成绩保存到文件中。 三、算法提示: 1、数据结构:结构体类型数组。 2、数据库结构:下表构成该系统的基本数据库。 姓名 学号 课程名称1 课程名称2 ●●●●●● char Char float float 四、测试数据: 学生人数N=10 课程门数M=4 课程名:数学、语文、英语、政治 五、其它 对该系统有兴趣的同学可以在实现上述基本功能后,完善系统的其它功能。 问题补充: 今天谁能给我答案啊!!!!!!!急急!!!!!!! 提问者: 342123465 - 试用期 一级 最佳答案 C语言课程设计报告—班级成绩管理系统 需要分析: 学生成绩管理系统有13种功能。把这13个功能做成13个子函数。在主函当数中设计一个菜单对这13个子数进行管理。来实现对整个系统的操作。 根据课题的要求。每一个学生的包括姓名(char)、学号(char)、M门课程的成绩(float).再加上系统功能上的要求每一学生的信息还要总分和名次等。所以自然的想到要用结构体来定义每一个学生的信息结构。然后用链表把它们组成一个有序的整体。用对链表的操作来实现对所有学生信息的统一管理(成绩显示、成绩排序、成绩修改等)。最后为了以后按照处理后的顺序保存到文件中。 . 各函数的功能: 概要设计: 程序的模块组成: 主 函 数: int main() 新建函数:STUDENT *init() 输入函数 :STUDENT *create() 显示函数: void print(STUDENT *head) 删除函数: STUDENT *delete(STUDENT *head) 按名字寻找函数: void lookup(STUDENT *head) 保存函数: void save(STUDENT *head) 按总分排序函数: STUDENT *sort(STUDENT *head) 计算总分和均分函数: void computer(STUDENT *h) 修改函数: STUDENT *Modify(STUDENT *head,STUDENT *new) 按学号排序函数: STUDENT *index(STUDENT *h) 菜单函数:int menu_select() 各个函数的主要功能: 输入函数: 随时输入数据。 菜单函数:显示系统主菜单。 显示函数: 显示所有学生的信息。 寻找函数: 方便学生查找自己的成绩。 删除函数: 删除某学生的信息。 排序函数: 按总成绩排序。 按学号排序函数: 按学号排序。 插入函数: 可以插入新的信息。 保存函数: 保存好学生成绩,以免丢失。 统计函数: l 显示每门课程成绩最高的学生的基本信息。 l 显示每门课程的平均成绩。 l 显示超过某门课程平均成绩学生人数。 课题的功能模块的划分: 开始 菜单界面 功能选择 初始化函数 输入学生信息 删除学生信息 显示学生信息 查找学生信息 按成绩排序 保存到文件 从文件读数据 插入学生成绩 分类合计 退出系统 结束 详细设计: 整个系统除了主函数外,另外还有14个函数,实现八大功能:输入功能、显示功能、查找功能、排序功能、插入功能、保存功能、读取功能。各个函数的详细设计说明分别如下: 主函数 main() 利用无限次循环for(;;)和swithch()实现各函数的调用,系统根据输入的数字选项来调用相应的函数。 菜单选择函数 int menu_select() 这是一个无参函数,主要实现“功能选择”的界面,在这个界面里有显示系统的九大功能,根据每个功能前面的序号进行选择。等执行完每一个函数功能后,返回菜单。 代码设计: 初始化函数 STUDENT *init() 这是一个无参函数,里面只有两个语句,它的作用是使链表初始化,使head的值为NULL和一个清屏语句。比如:没有这个函数的话,在你没有输入任何数据的情况下,去执行显示功能的时候会显示一些乱

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值