【C语言】学生管理系统:完整模拟与实现

## 标题

请添加图片描述
Alt

🌈个人主页:是店小二呀
🌈C语言笔记专栏:C语言笔记
🌈C++笔记专栏: C++笔记

🌈喜欢的诗句:无人扶我青云志 我自踏雪至山巅
请添加图片描述
🔥引言

本篇文章为修改了在校期间实训报告,使用C语言实现学生成绩管理系统。对此,其中步骤没有详细写出,如果有问题可以私信我,感谢你的支持。

文章目录

  • 一、背景描述
  • 二、任务需求
  • 三、总体设计
    • 3.1开放平台
    • 3.2 总体思路
  • 四、功能模板设计:
    • 4.1 模拟实现顺序表
      • 4.1.1 顺序表的基本结构
      • 4.1.2 顺序表的初始化
      • 4.1.3 顺序表的销毁
      • 4.1.4顺序表的扩容(为插入数据提供保障)
      • 4.1.5 顺序表的尾插
      • 4.1.6 顺序表的判空
      • 4.1.7 顺序表的任意位置删除(pos是下标)
    • 4.2 实现学生成绩管理系统
      • 4.2.1 学生成绩管理系统需要实现的接口
      • 4.2.2 typedef重定义类型的名字
    • 4.3 实现学生成绩管理系统接口(主要是对顺序表接口的复用)
      • 4.3.1学生信息的初始化
      • 4.3.2 学生信息的销毁
      • 4.3.3 添加学生信息
      • 4.3.4 查找指定学生的下标
      • 4.3.5 删除学生信息
      • 4.3.6 查看学生成绩信息
      • 4.3.7 修改学生信息
      • 4.3.8 查找指定学生信息
      • 4.3.9 按照名字或者成绩排序
      • 4.3.10 比较函数的接口
    • 4.4 菜单界面
  • 五、以下是系统测试情况


一、背景描述

学生成绩管理系统是用于存储学生个人信息,对于学生信息进行系统的管理。关于学生成绩管理系统,不单单只能适用于学生信息,该系统的底层逻辑,同样适用于需要多个对对象复杂信息进行存储和管理的场景。

二、任务需求

对于学生成绩管理系统,需要设计以下接口功能,才能保证系统的基本运行和提高系统的可维护性。接口:学生信息录入、输出、查询、修改,排序等功能。包括系统的控制面板,通过输入控制对应接口的调用。

三、总体设计

3.1开放平台

本次学生成绩管理系统在DEV-C++轻量级编译器下实现,并且通过C语言编写该程序。

3.2 总体思路

我们将通过该系统的底层逻辑针对性的实现接口。学生信息是具有复杂的信息,需要使用结构体(类)进行封装这些信息,而对于多个学生对象需要使用数组进行存储,但是数组的大小在编译阶段就被确定,属于静态数组。对于数组的大小无法合理的分配,大小给大了导致浪费,开小了又不够使用。对此,需要使用动态开辟内存的数据结构来存在我们学生的数据,这种数据结构称为顺序表。

在这里插入图片描述
对于,实现学生成绩管理系统就需要借用顺序表的结构和接口。对此我们将学生成绩管理系统分为两大过程:模拟实现顺序表,以管理系统为目标对顺序表进行应用
在这里插入图片描述
学生成绩管理系统是基于顺序表的应用,对此需要先实现顺序表或者使用STL,根据管理系统的要求进行处理。
在这里插入图片描述

四、功能模板设计:

功能模板根据两个大过程:模拟实现顺序表,顺序表的应用实现管理系统。

4.1 模拟实现顺序表

4.1.1 顺序表的基本结构

#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#include <stdbool.h>
#include <Windows.h>typedef struct AchievementInfo SLDataType;
//顺序表的基本结构
typedef struct SeqList
{SLDataType* _a;int _size; //顺序表中有效元素int _capacity; //顺序表当前容量
}SL;

4.1.2 顺序表的初始化

void SLInit(SL* ps)
{assert(ps);ps->_a = NULL;ps->_size = 0;//可以选择给数据或者不给//这先不给,扩容需要_a指向一个明确的空间ps->_capacity = 0;
}

4.1.3 顺序表的销毁

void SLDestroy(SL* ps){assert(ps);assert(ps->_a);//释放动态开辟内存free(ps->_a);ps->_a = NULL;ps->_size = 0;ps->_capacity = 0;}

4.1.4顺序表的扩容(为插入数据提供保障)

void SLCheckCapacity(SL* ps)
{assert(ps);if (ps->_capacity == ps->_size){int new_cpacity = ps->_capacity == 0 ? 4 : ps->_capacity * 2;SLDataType* tmp = (SLDataType*)realloc(ps->_a, sizeof(SLDataType) * new_cpacity);if (tmp == NULL){perror("realloc fail!");return ;}ps->_a = tmp;ps->_capacity = new_cpacity;}
}

在实现该接口时】:

  • 为存储数据而申请的一块空间,那么需要交给这个数据类型去维护

  • Capacity代表当前空间大小,Size代表当前有效数据,当有效数据充满了当前空间大小就需要申请内存空间(这里需要实现多次插入函数,这里单独实现SLChekckCapacity)

  • newcapacity是防止在扩容时,capacity为空(零乘任何数为零),申请空间大小错误

  • 最好不要phead直接接收扩容的地址,防止扩容(第二种情况)失败导致找不到之前空间地址

  • 开辟以字符类型来维修开辟的空间,需要为‘\0‘开辟一个空间

4.1.5 顺序表的尾插

//顺序表的尾插
void SLPushBack(SL* ps, SLDataType x)
{assert(ps);if (ps->_capacity == ps->_size) SLCheckCapacity(ps);ps->_a[ps->_size++] = x;
}

4.1.6 顺序表的判空

bool SLEmpty(SL*ps)
{assert(ps);assert(ps->_a);return ps->_size==0;
}

4.1.7 顺序表的任意位置删除(pos是下标)

//任意位置删除void SLErase(SL* ps, int pos){assert(!(SLEmpty(ps)));assert(ps);assert(ps->_a);assert(0 <= pos && pos < ps->_size);for (int i = pos; i < ps->_size; i++){ps->_a[i] = ps->_a[i + 1];}ps->_size--;}

在实现该接口时】:

  • 需要对pos设置范围

  • 以下标pos为界,pos之后的数据向前移动(跟头删是类似的,主要是在循环条件略显差异)

4.2 实现学生成绩管理系统

首先学生成绩管理系统是在顺序表数据结构的基础上,进行灵活的应用,对此需要包括顺序表的头文件,便于调用顺序表中实现的接口。

4.2.1 学生成绩管理系统需要实现的接口

以下是Management System.h头文件,主要用于定义学生信息的结构和该系统需要实现的接口

#define NAME_MAX 100
#define SEX_MAX 10
#define REGISTRATION_MAX 30
#define Grades_MAX 10enum AcInfo
{Name = 1,Registration,Grades
};struct AchievementInfo
{//学生姓名char _name[NAME_MAX];//学生性别char _sex[SEX_MAX];//学生学籍号char _registration[REGISTRATION_MAX];//学生成绩int _grades;
};typedef struct AchievementInfo AInfo; typedef struct SeqList Achievement;
//学生信息的初始化
void Achievement_Init(Achievement* ac);
//学生信息的销毁
void Achievement_Destroy(Achievement* ac);//添加学生信息
void Achievement_Add(Achievement* ac);
//删除学生信息
void Achievement_Del(Achievement* ac);
//修改学生成绩信息
void Achievement_Modify(Achievement* ac);
//查看全部学生信息
void Achievement_Show(Achievement* ac);
查找指定学生信息
void  Achievement_Find(Achievement* ac);
//按照名字或者成绩排序
void Achievement_Sort(Achievement* ac);

4.2.2 typedef重定义类型的名字

//对于顺序表结构体类型重定义类型typedef struct SeqList AInfo;//对于顺序表内嵌结构体重定义类型typedef struct AchievementInfo AInfo;

4.3 实现学生成绩管理系统接口(主要是对顺序表接口的复用)

在这里插入图片描述

4.3.1学生信息的初始化

void Achievement_Init(Achievement* ac)
{SLInit(ac);
}

4.3.2 学生信息的销毁

void Achievement_Destroy(Achievement* ac)
{SLDestroy(ac);
}

4.3.3 添加学生信息

void Achievement_Add(Achievement* ac)
{AInfo info;printf("请分别输入学生的名字、性别、学号、成绩\n");scanf("%s %s %s %d", info._name, info._sex, info._registration, &info._grades);//往(顺序表)中插入数据SLPushBack(ac, info);
}

4.3.4 查找指定学生的下标

int FindSTName(Achievement* ac, char name[])
{for (int i = 0; i < ac->_size; i++){if (strcmp(ac->_a[i]._name, name) == 0 ){return i;}}return -1;
}

4.3.5 删除学生信息

void Achievement_Del(Achievement* ac)
{assert(ac);//根据用户的名字进行删除printf("请输入你需要删除的学生姓名\n");char name[NAME_MAX];scanf("%s", name);int findidex = FindSTName(ac, name);if (findidex < 0){printf("你需要删除的学生信息不存在\n");return;}//找到了进行删除操作SLErase(ac, findidex);
}

4.3.6 查看学生成绩信息

void Achievement_Show(Achievement* ac)
{printf("系统正在加载中....\n");Sleep(3000);printf("系统加载完成!\n");//打印表头信息printf("%s %s %-10s %s\n", "学生姓名", "学生性别", "学生学号", "学生成绩");for(int i =0; i < ac->_size; i++){printf("%-10s %-5s %-8s %-d分\n", ac->_a[i]._name, ac->_a[i]._sex, ac->_a[i]._registration, ac->_a[i]._grades);}
}

4.3.7 修改学生信息

void Achievement_Modify(Achievement* ac)
{assert(ac);//根据用户的名字进行修改 成绩printf("请输入你需要修改的学生姓名\n");char name[NAME_MAX];scanf("%s", name);int findidex = FindSTName(ac, name);if (findidex < 0){printf("你需要修改的学生信息不存在\n");return;}printf("系统正在加载中....\n");Sleep(3000);printf("系统加载完成!\n");printf("请重新分别输入学生的名字、性别、学号、成绩\n");scanf("%s %s %s %d", ac->_a[findidex]._name, ac->_a[findidex]._sex,ac->_a[findidex]._registration,&ac->_a[findidex]._grades);printf("修改成功!\n");
}

4.3.8 查找指定学生信息

void  Achievement_Find(Achievement* ac)
{assert(ac);//根据用户的名字进行修改 成绩printf("请输入你需要查找的学生姓名\n");char name[NAME_MAX];scanf("%s", name);int findidex = FindSTName(ac, name);if (findidex < 0){printf("你需要查找的学生信息不存在\n");return;}printf("系统正在加载中....\n");Sleep(3000);printf("系统加载完成!\n");printf("以下是你需要查找的学生信息\n");printf("%-10s %-5s %-8s %-d分\n", ac->_a[findidex]._name, ac->_a[findidex]._sex,ac->_a[findidex]._registration, ac->_a[findidex]._grades);
}

4.3.9 按照名字或者成绩排序

//按照名字或者成绩排序void Achievement_Sort(Achievement* ac){enum AcInfo select;printf("请输入你需要按照什么类型排序:(1->Name,2->Registration,3->Grades)\n");// 清空输入缓冲区fflush(stdin);scanf("%u", &select);if (select < Name || select > Grades){printf("输入的排序类型无效!\n");return; // 或者采取其他合适的处理方式}//这个名字就代表什么数据switch (select){case Name:qsort(ac->_a, ac->_size, sizeof(ac->_a[0]), Name_Compare);break;case Registration:qsort(ac->_a, ac->_size, sizeof(ac->_a[0]), Registration_Compare);break;case Grades:qsort(ac->_a, ac->_size, sizeof(ac->_a[0]), Grades_Compare);break;default:break;}printf("排序成功\n");}

4.3.10 比较函数的接口

int Name_Compare(const void* e1, const void* e2)
{//是每个元素之间的比较const AInfo* a1 = (const AInfo*) e1;const AInfo* a2 = (const AInfo*) e2;return strcmp(a1->_name, a2->_name);
}int Registration_Compare(const void* e1, const void* e2)
{//是每个元素之间的比较const AInfo* a1 = (const AInfo*)e1;const AInfo* a2 = (const AInfo*)e2;return strcmp(a1->_registration, a2->_registration);
}
int Grades_Compare(const void* e1, const void* e2)
{//是每个元素之间的比较const AInfo* a1 = (const AInfo*)e1;const AInfo* a2 = (const AInfo*)e2;//如果是100分就会出现问题//是根据字符的大小进行判断//所以这里成绩可以整型的比较进行return a1->_grades - a2->_grades;
}

4.4 菜单界面

#define _CRT_SECURE_NO_WARNINGS 1
#include "SeqList.h"
#include "Management System.h"void mune()
{printf("*****************************************************\n");printf("************欢迎使用学生成绩管理系统*****************\n");printf("*****1.添加学生信息********2.删除学生信息************\n");printf("*****************************************************\n");printf("*****3.修改学生信息********4.查找指定学生信息*********\n");printf("*****************************************************\n");printf("*****5.查看全部学生信息****6.按照名字或者成绩排序******\n");printf("***************0.退出系统*****************************\n");printf("*****************************************************\n");}int main()
{typedef struct SeqList AInfo;AInfo ac;Achievement_Init(&ac);int input;mune();do{printf("请根据菜单选择你需要完成的操作\n");// 清空输入缓冲区fflush(stdin);scanf("%d", &input);printf("请稍等!\n");switch (input){case 0: printf("成功退出该系统");Achievement_Destroy(&ac);break;case 1: Achievement_Add(&ac);break;case 2: Achievement_Del(&ac);break;case 3: Achievement_Modify(&ac);break;case 4: Achievement_Find(&ac);break;case 5: Achievement_Show(&ac);break;case 6: Achievement_Sort(&ac);break;default:printf("非法输入,请重新输入\n");break;}} while (input);return 0;
}

五、以下是系统测试情况

在这里插入图片描述
在这里插入图片描述
将sqort比较函数是对于元素进行比较,在强转类型转化的时候,类型我给了定义顺序表结构的结构体类型,而不是顺序表中内嵌学生信息的结构体类型,所以导致了错误。

当然这一块学生按照名字,学号,成绩排序,在学习枚举时。我想到了以枚举类型代替数据,从而配合switch分支语句,进行选择性的根据不同要求进行排序,这也是属于我比较满意的地方。
在这里插入图片描述


请添加图片描述

以上就是本篇文章的所有内容,在此感谢大家的观看!这里是店小二C语言笔记,希望对你在学习C语言中有所帮助!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/367503.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【网络架构】lvs集群

目录 一、集群与分布式 1.1 集群介绍 1.2 分布式系统 1.3 集群设计原则 二、LVS 2.1 lvs工作原理 2.2 lvs集群体系架构 ​编辑 2.3 lvs功能及组织架构 2.4 lvs集群类型中术语 三、LVS工作模式和命令 3.1 lvs集群的工作模式 3.1.1 lvs的nat模式 3.1.2 lvs的dr模式 …

为什么有些人思考得多,决策反而不好?避免过度拟合的终极指南:决策高手的秘密:灰度认知,黑白决策

在决策过程中&#xff0c;过度关注细节可能导致决策效果不佳&#xff0c;这被称为“过度拟合”。为了避免这种情况&#xff0c;我们需要进行“灰度认知&#xff0c;黑白决策”&#xff0c;即接受不确定性&#xff0c;关注整体趋势&#xff0c;设定明确目标&#xff0c;简化选择…

【JD-GUI】MacOS 中使用Java反编译工具JD-GUI

希望文章能给到你启发和灵感&#xff5e; 如果觉得文章对你有帮助的话&#xff0c;点赞 关注 收藏 支持一下博主吧&#xff5e; 阅读指南 开篇说明概念理解一、基础环境说明1.1 硬件环境1.2 软件环境 二、下载与安装2.1 选择对应版本2.2 解压运行排除异常&#xff1a;2.3 关于…

Geotools系列说明之LineString仿高德航路截取说明

需求分析 我们在做webgl的时候经常会遇到这样的需求&#xff0c;计算给定航路的拥堵情况&#xff0c;不同的拥堵显示不同的颜色&#xff0c;航路截取计算等等。基于这类问题统一都可以使用LineString进行处理 实现思路 如上图所示&#xff0c;航路是几个关键的点然后练成线&a…

05 docker 镜像

目录 1. 镜像 2. 联合文件系统 3. docker镜像加载原理 4. 镜像分层 镜像分层的优势 5. 容器层 1. 镜像 镜像是一种轻量级、可执行的独立软件包&#xff0c;它包含运行某个软件所需的所有内容&#xff0c;我们把应用程序和配置依赖打包好行程一个可交付的运行环境&#xf…

监控平台zabbix对接grafana

目录 1.安装grafana并启动 2.浏览器访问 3.导入zabbix数据&#xff0c;对接grafana 4.如何导入模板 5.使用zabbix监控nginx并发量连接数 5.1 修改nginx配置 5.2 编写监控数据脚本 5.3 设置键值 5.4 在zabbix web端完成自定义监控项 5.5 连接到grafana 以上一篇博客&l…

Python 生成Md文件带超链 和 PDF文件 带分页显示内容

software.md # -*- coding: utf-8 -*- import os f open("software.md", "w", encoding"utf-8") f.write(内部测试版2024 MD版\n) for root, dirs, files in os.walk(path): dax os.path.basename(root)if dax "":print("空白…

基于YOLOv9的PCB板缺陷检测

数据集 PCB缺陷检测&#xff0c;我们直接采用北京大学智能机器人开放实验室数据提供的数据集&#xff0c; 共六类缺陷 漏孔、鼠咬、开路、短路、杂散、杂铜 已经对数据进行了数据增强处理&#xff0c;同时按照YOLO格式配置好&#xff0c;数据内容如下 模型训练 ​ 采用YOLO…

StarRocks 3.3 重磅发布,Lakehouse 架构发展进入快车道!

StarRocks 3.3 的发布标志着 Lakehouse 架构在数据分析领域迈向了一个新的高度。作为下一代 Lakehouse 架构的代表&#xff0c;StarRocks 3.3 在稳定性、计算性能、缓存设计、物化视图、存储优化和 Lakehouse 生态系统等方面进行了全方位的优化和创新。本文将逐一介绍 StarRock…

如何在 SQL 中删除一条记录?

如何在 SQL 中删除一条记录&#xff1f; 在 SQL 中&#xff0c;您可以使用DELETE查询和WHERE子句删除表中的一条记录。在本文中&#xff0c;我将向您介绍如何使用DELETE查询和WHERE子句删除记录。我还将向您展示如何一次从表中删除多条记录 如何在 SQL 中使用 DELETE 这是使…

ctfshow sql注入 web234--web241

web234 $sql "update ctfshow_user set pass {$password} where username {$username};";这里被过滤了&#xff0c;所以我们用\转义使得变为普通字符 $sql "update ctfshow_user set pass \ where username {$username};";那么这里的话 pass\ where…

踩坑:Unity导出WebGL发布到手机上竖屏时强制显示横屏

具体的适配问题 公司的项目需要将游戏导出WebGL 发布到Web平台 本以为是个很简单的事情 谁知道却被个横竖屏适配搞的头晕 毕竟只有大学浅浅的学了下HTML这门语言 出来工作后基本上都是在跟C# Lua打交道 言归正传 看看具体问题吧 游戏如果从横屏进入 基本上不会有什么适配问题…

预处理详解

1.预定义符号 C语言设置了一些预定义符号&#xff0c;可以直接使用&#xff0c;预定义符号也是在预处理期间处理的。 __FILE__ //进行编译的源文件 __LINE__ //文件当前的行号 __DATE__ //文件被编译的日期 __TIME__ //文件被编译的时间 __STDC__ //如果编译器遵循ANSI C&…

在Clion使用CubeMX Stm32的步骤

Step1 准备软件&#xff0c;安装环境&#xff1a; 1. cubemx v6.5.0&#xff08;可以兼容以前版本的project&#xff09; https://www.st.com.cn/zh/development-tools/stm32cubemx.html STM32CubeMX 默认安装目录, 6.5版本可以兼容老版本 C:\Program Files\STMicroelectroni…

SpringMVC常见的注解

一、Spring MVC Spring Web MVC是基于ServletAPI构建的原始web 框架&#xff0c;一开始就包含在Spring 框架中&#xff0c;通常被称为“Spring MVC”。 1.MVC 是什么&#xff1f; MVC(Model、View、Controller&#xff09;是软件工程中的一种软件架构设计模型。它把软件系统分…

Vision Transformer论文阅读笔记

目录 An Image is Worth 16x16 Words: Transformers for Image Recognition at Scale -- Vision Transformer摘要Introduction—简介RELATED WORK—相关工作METHOD—方法VISION TRANSFORMER (VIT)—视觉Transformer(ViT) 分析与评估PRE-TRAINING DATA REQUIREMENTS—预训练数据…

战略流程-麦肯锡企业数字化业务变革成熟度评估模型及案例深度解析

一、企业变革成熟度评估模型 企业变革成熟度诊断模型是一种评估工具&#xff0c;用于全面扫描和评估企业在变革转型过程中的能力水平。该模型通过一系列量化指标和定性分析&#xff0c;对企业在不同变革领域的成熟度进行评分&#xff0c;从而帮助企业识别在变革过程中的优势和…

llm学习-3(向量数据库的使用)

1&#xff1a;数据读取和加载 接着上面的常规操作 加载环境变量---》获取所有路径---》加载文档---》切分文档 代码如下&#xff1a; import os from dotenv import load_dotenv, find_dotenvload_dotenv(find_dotenv()) # 获取folder_path下所有文件路径&#xff0c;储存在…

mysql数据库自动备份

crond cron 实现定时执行 安装 yum install crond启动 service crond start查看状态 service crond status执行 crontab xx.cron查看任务列表 crontab -l删除所有任务 crontab -r 示例&#xff1a;每分钟写入 Good morning 到 mine.cron [rootecs-f1dd-0001 fztmp]# echo ‘* …

什么是 API 代理?

API 代理就像是您的计算机和互联网上特殊服务之间的中间人。它有点像集翻译、保安和信使于一体。 什么是 API 代理&#xff1f; API 代理就像是您和在线服务之间的中间人。当您的计算机需要从某个特殊的在线服务 (API) 获得某些东西时&#xff0c;API 代理会确保一切顺利进行…