网站制作需要注意什么/国内新闻大事20条简短
🌈个人主页:是店小二呀
🌈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语言中有所帮助!
相关文章:

【C语言】学生管理系统:完整模拟与实现
🌈个人主页:是店小二呀 🌈C语言笔记专栏:C语言笔记 🌈C笔记专栏: C笔记 🌈喜欢的诗句:无人扶我青云志 我自踏雪至山巅 🔥引言 本篇文章为修改了在校期间实训报告,使用C…...

pypi 发布自己的包
注册pypi个人用户 网址:https://pypi.org 目录结构dingtalk_utils 必须-pkgs- __init__.py .gitignore LICENSE 必须 README.md 必须 requirements.txt setup.py 必须安装依赖 pip install setuptools wheel安装上传工具 pip install twinesetup.py i…...

关闭windows11磁盘地址栏上的历史记录
关闭windows11的磁盘地址栏上的历史记录 windows11打开磁盘后访问某一个磁盘路径后会记录这个磁盘路径,而且有时候会卡住这个地址栏(关都关不掉),非常麻烦。 如下图所示: 关闭地址栏历史记录 按下windows键打开开…...

DDS自动化测试落地方案 | 怿星科技携最新技术亮相是德科技年度盛会
5月28日,怿星科技作为是德科技的重要合作伙伴亮相Keysight World Tech Day 2024。在此次科技盛会上,怿星科技不仅展示了领先的DDS自动化测试解决方案等前沿技术,还分享了在“周期短、任务重”的情况下,如何做好软件开发和测试验证…...

新品!和芯星通全系统全频高精度板卡UB9A0首发
6月6日,和芯星通发布了UB9A0全系统全频高精度GNSS板卡,主要应用于CORS站、便携基站、GNSS全球监测跟踪站等。延续了上一代产品高质量原始观测量的特点,UB9A0在性能和稳定性方面均表现出众。 UB9A0基于射频基带及高精度算法一体化的GNSS SoC芯…...

Cognita RAG:模块化、易用与可扩展的开源框架
Cognita RAG是一个开源框架,它通过模块化设计、用户友好的界面和可扩展性,简化了将领域特定知识整合到通用预训练语言模型中的过程。本文介绍了Cognita的特点、优势、应用场景以及如何帮助开发者构建适合生产环境的RAG应用程序。 文章目录 Cognita RAG介…...

linux虚拟机免密登录配置
1、假设A服务器要免密登录B服务器 2、在A服务器上执行命令: cd /root/.ssh/ ssh-keygen -t rsa #这里会生成两个文件 一个是id_rsa私钥和公钥rsa.pub2、我们把公钥的内容复制粘贴到B服务器的/root/.ssh/authorized_keys文件下 #在A服务器上执行命令记录内容 cat …...

Qt_C++ RFID网络读卡器Socket Udp通讯示例源码
本示例使用的设备: WIFI/TCP/UDP/HTTP协议RFID液显网络读卡器可二次开发语音播报POE-淘宝网 (taobao.com) #ifndef MAINWINDOW_H #define MAINWINDOW_H#include <QMainWindow> #include <QHostInfo> #include <QNetworkInterface> #include <…...

C++ 实现Python 列表list 的两种方法
1、vector里面放多种参数。在C中,如果你想要在std::vector中存储不同类型的参数,你可以使用std::any(C17及以上)或std::variant(C17以前的版本需要使用Boost库或者C17及以上标准)。以下是使用std::vector&l…...

vue3+ elementPlus PC端开发 遇到页面已进入就form校验了的问题
form表单一进页面就校验了 rules里配置的 require 提示语 如图所示代码是这样的 最后发现是form表单下面的一个按钮的展示规则 会导致规则校验 canAddInsured 这个字段的变化会导致form表单校验 这个字段是computed maxInsureds 也是个computed监听 maxInsured.value >1 就…...

transformers DataCollator介绍
本博客主要介绍 transformers DataCollator的使用 from transformers import AutoTokenizer, AutoModel, \DataCollatorForSeq2Seq, DataCollatorWithPadding, \DataCollatorForTokenClassification, DefaultDataCollator, DataCollatorForLanguageModelingPRETRAIN_MODEL &qu…...

rust学习(字节数组转string)
最新在写数据传输相关的操作,发现string一个有趣的现象,代码如下: fn main() {let mut data:[u8;32] [0;32];data[0] a as u8;let my_str1 String::from_utf8_lossy(&data);let my_str my_str1.trim();println!("my_str len is…...

Docker:技术架构演进
文章目录 基本概念架构演进单机架构应用数据分离架构应用服务集群架构读写分离/主从分离架构冷热分离架构垂直分库微服务容器编排架构 本篇开始进行对于Docker的学习,Docker是一个陌生的词汇,那么本篇开始就先从技术架构的角度出发,先对于技术…...

汽车MCU虚拟化--对中断虚拟化的思考(2)
目录 1.引入 2.TC4xx如何实现中断虚拟化 3.小结 1.引入 其实不管内核怎么变,针对中断虚拟化无非就是上面两种,要么透传给VM,要么由Hypervisor统一分发。汽车MCU虚拟化--对中断虚拟化的思考(1)-CSDN博客 那么,作为车规MCU龙头…...

python的继承
本章正式开始之前,先让我们回顾一下什么是 对象 ? 什么是 类 ? 小贝 喜欢 猫咪,今年领养了一只名叫 Kitty 的 布偶猫。则下列哪项是 对象 呢? A. 猫咪 B. Kitty C. 布偶猫 相比之下,闻闻 更喜欢 犬科 动…...

组件的注册和引用
在Vue中,开发者可以将页面中独立的、可重用的部分封装成组件,对组件的结构,样式和行为进行设置。组件是 Vue 的基本结构单元,组件之间可以相互引用。 一.注册组件 当在Vue项目中定义了一个新的组件后,要想在其他组件中…...

诊所如何赢得患者?做好这两点很关键!
大家都知道,社区周边的诊所原本是居民看病的第一选择,方便又快捷。但现在很多诊所服务都差不多,没有自己的特色,这就让患者有点难选择了。那诊所怎么做才能更吸引患者呢?其实,关键是要抓住患者的心…...

Qwen2本地部署的实战教程
大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算法…...

html+CSS+js部分基础运用15
1、完成输入框内容的实时反向输出。 2、银行账户余额变动自动通知项目。 设计要求:单击按钮后,余额按照输入框的数额减少,同时将按钮式的提示信息(金额)同步改变。利用侦听属性实现余额发生变化时发出提示信息&#x…...

从零开始学JAVA
一、编写Hello world程序 public class JavaMain1 {//主程序执行入口,main方法public static void main(String[] args){System.out.println("Hello world!");} } 运行结果 Hello world! java编写主程序常见错误: 1、System ---首字母没有…...

MySQL(四)查询
1、MySQL限性约束 —非空、唯一(自增)、主外键、检查(MySQL存在但是不能用)。 约束主要完成对数据的校验,保证数据库数据的完整性;如果有相互依赖数据,保证该数据不被删除。 1)常用五类约束 not null :非空约束,指定某列不为空。 unique:唯一约束,指定某列和几列组…...

嵌入式学习——网络编程(TCP)——day31
1. TCP和UDP的区别 TCP(Transmission Control Protocol,传输控制协议) UDP(User Datagram Protocol,用户数据报协议) 1.1 连接方式 TCP 是面向连接的协议,它在数据传输前需要通过三次握手建立…...

[STM32]定位器与PWM的LED控制
目录 1. 深入了解STM32定时器原理,掌握脉宽调制pwm生成方法。 (1)STM32定时器原理 原理概述 STM32定时器的常见模式 使用步骤 (2)脉宽调制pwm生成方法。 2. 实验 (1)LED亮灭 代码 测试效果 (2)呼吸灯 代码 测试效果 3.总结 1. 深入了解STM32定时器原…...

可视化数据科学平台在信贷领域应用系列五:零代码可视化建模
信贷风控模型是金融机构风险管理的核心工具,在信贷风险管理工作中扮演着至关重要的角色。随着信贷市场的环境不断变化,信贷业务的风险日趋复杂化和隐蔽化,开发和应用准确高效的信贷风控模型显得尤为重要。信贷风险控制面临着越来越大的挑战和…...

Windows 11广告植入“另辟蹊径”:PC Manager暗示若不使用必应搜索,你的系统可能需要“修复”
Edge浏览器近期增添了许多实用的新功能,如侧边栏、休眠标签页和沉浸式阅读器。话虽如此,浏览器中仍有一部分功能被部分用户视为“冗余软件”和不必要的累赘。 随着Windows 11用户逐渐习惯操作系统关键位置出现越来越多的广告,微软似乎正尝试以…...

一线教师教学工具汇总
亲爱的教师们!我们的教学工具箱里也该更新换代啦!今天,就让我来给大家安利一波超实用的教学神器: 百度文库小程序 —— 在线图书馆 百度文库,一个宝藏级的在线文档分享平台!在这里,你可以找到海…...

【数据结构】栈和队列-->理解和实现(赋源码)
Toc 欢迎光临我的Blog,喜欢就点歌关注吧♥ 前面介绍了顺序表、单链表、双向循环链表,基本上已经结束了链表的讲解,今天谈一下栈、队列。可以简单的说是前面学习的一特殊化实现,但是总体是相似的。 前言 栈是一种特殊的线性表&…...

一篇教会你CSS定位
前言:在网页布局的时候,我们需要将想要的元素放到指定的位置上,这个时候我们就可以使用CSS中的定位操作。 先让我们看一下本篇文章的大致内容: 目录 什么是定位 1.相对定位 2.绝对定位 3. 固定定位 4. 粘性定位 5. 定位层级…...

Hive的常规操作
Hive常规操作 hive常用交互命令 -e执行sql语句 [rootmaster ~]# hive -e "show databases";-f执行sql脚本 [rootmaster ~]# hive -f /usr/local/demo.sql查看hive中输入的所有命令 [rootmaster ~]# cat ~/.hivehistory操作库 创建库 语法: create…...

redis做为缓存,mysql的数据如何与redis进行同步呢?
让我们一步步来实现如何让MySQL数据库的数据和Redis缓存保持同步。想象一下,MySQL是一个大仓库,存放着所有重要的货物(数据),而Redis则像是一个快速取货窗口,让你能更快拿到常用的东西。为了让两者保持一致…...