c++ 实战案例--学生管理系统

简介: c++ 实战案例--学生管理系统

需求

1、学生信息的添加;

2、计算每个学生的总分;

3、输出所有学生的信息;

4、按照总分排序输出学生信息;

5、按照姓名查询学生情况;

6、按照学号删除某个学生,并输出删除后的所有学生信息;

7、将所有学生的信息存储在文本文件“student.txt”中。

8、从文件中读出信息。(打开文件)

9、退出程序

源码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
// 定义结构体表示学生信息
typedef struct student
{
    long studentID;
    char name[20];
    char sex[2];
    float chinese;   // 语文
    float maths;     // 数学
    float chemistry; // 化学
    float physics;   // 物理
    float English;   // 英语
    float sumScore;  // 总分
} STUDENT;
void addStudent(STUDENT students[], int* count);                         // 添加学生信息
void calculateSum(STUDENT students[], int count);                        // 计算每个学生的总分
void displayStudent(STUDENT students[], int count);                      // 显示学生信息
void writeStudent(STUDENT students[], int count);                        // 将学生信息写到student.txt中
void readStudent(STUDENT students[], int* count);                        // 从student.txt中读出学生信息
void sortStudentsByTotalScore(STUDENT students[], int count);            // 按总分排序学生信息
void findStudentByName(STUDENT students[], int count, const char* name); // 按姓名查找学生信息
void deleteStudentByID(STUDENT students[], int* count, long studentID);  // 按学号删除学生信息
int main()
{
    STUDENT students[100];
    int count = 0;
    while (1)
    {
        printf("\n\t\t\t\t请选择操作:\n");
        printf("\t\t\t\t1. 打开文件\n");
        printf("\t\t\t\t2. 添加学生信息\n");
        printf("\t\t\t\t3. 计算学生总分\n");
        printf("\t\t\t\t4. 显示学生信息\n");
        printf("\t\t\t\t5. 查找学生\n");
        printf("\t\t\t\t6. 删除学生\n");
        printf("\t\t\t\t7. 按总分降序输出\n");
        printf("\t\t\t\t8. 保存文件\n");
        printf("\t\t\t\t9. 退出\n");
        int choice;
        scanf("%d", &choice);
        switch (choice)
        {
        case 1:
            readStudent(students, &count);
            break;
        case 2:
            addStudent(students, &count);
            break;
        case 3:
            calculateSum(students, count);
            break;
        case 4:
            displayStudent(students, count);
            break;
        case 5:
        {
            char name[20];
            printf("请输入要查找的学生姓名:");
            scanf("%s", name);
            findStudentByName(students, count, name);
        }
        break;
        case 6:
        {
            long studentID;
            printf("请输入要删除的学生学号:");
            scanf("%ld", &studentID);
            deleteStudentByID(students, &count, studentID);
            displayStudent(students, count);
        }
        break;
        case 7:
            sortStudentsByTotalScore(students, count);
            displayStudent(students, count);
            break;
        case 8:
            writeStudent(students, count);
            break;
        case 9:
            exit(0);
        default:
            printf("无效的选择,请重新选择!\n");
            break;
        }
    }
    return 0;
}
void addStudent(STUDENT students[], int* count)
{
    STUDENT student;
    printf("请输入学生学号:");
    scanf("%ld", &student.studentID);
    printf("请输入学生姓名:");
    scanf("%s", student.name);
    printf("请输入学生性别:");
    scanf("%s", student.sex);
    printf("请输入学生语文成绩:");
    scanf("%f", &student.chinese);
    printf("请输入学生数学成绩:");
    scanf("%f", &student.maths);
    printf("请输入学生化学成绩:");
    scanf("%f", &student.chemistry);
    printf("请输入学生物理成绩:");
    scanf("%f", &student.physics);
    printf("请输入学生英语成绩:");
    scanf("%f", &student.English);
    students[*count] = student; // 将增加的学生信息添加到数组中
    (*count)++;                 // 学生数增加一个
}
// 显示学生信息
void displayStudent(STUDENT students[], int count)
{
    printf("\t\t学号\t姓名\t性别\t语文\t数学\t化学\t物理\t英语\t总分\n");
    for (int i = 0; i < count; i++)
    {
        printf("\t\t%ld\t%s\t%s\t%.2f\t%.2f\t%.2f\t%.2f\t%.2f\t%.2f\t\n",
            students[i].studentID, students[i].name, students[i].sex,
            students[i].chinese, students[i].maths, students[i].chemistry,
            students[i].physics, students[i].English, students[i].sumScore);
    }
}
// 计算每个学生的总分
void calculateSum(STUDENT students[], int count)
{
    for (int i = 0; i < count; i++)
    {
        students[i].sumScore = students[i].chinese + students[i].maths + students[i].chemistry + students[i].physics + students[i].English;
    }
}
void writeStudent(STUDENT students[], int count) // 将学生信息写到student.txt中
{
    FILE* fp = fopen("student.txt", "w");
    int i;
    if (fp == NULL)
    {
        printf("打开文件失败!\n");
        return;
    }
    fwrite(students, sizeof(STUDENT), count, fp);
    fclose(fp); // 如果没有这个语句,保存不了
}
void readStudent(STUDENT students[], int* count)
{
    FILE* fp = fopen("student.txt", "r");
    STUDENT student;
    if (fp == NULL)
    {
        printf("打开文件失败!\n");
        // exit(0);
        return;
    }
    while (!feof(fp))
    {
        fread(&students[*count], sizeof(STUDENT), 1, fp); /* 按数据块读文件 */
        (*count)++;
    }
    (*count)--;
    // printf("个数为:%d\n",*count);
    fclose(fp);
}
// 按照总分排序输出学生信息
void sortStudentsByTotalScore(STUDENT students[], int count)
{
    for (int i = 0; i < count - 1; i++)
    {
        for (int j = 0; j < count - 1 - i; j++)
        {
            if (students[j].sumScore < students[j + 1].sumScore)
            {
                STUDENT temp = students[j];
                students[j] = students[j + 1];
                students[j + 1] = temp;
            }
        }
    }
}
void findStudentByName(STUDENT students[], int count, const char* name)
{
    int found = 0;
    printf("\t\t学号\t姓名\t性别\t语文\t数学\t化学\t物理\t英语\t总分\n");
    for (int i = 0; i < count; i++)
    {
        if (strcmp(students[i].name, name) == 0)
        {
            printf("\t\t%ld\t%s\t%s\t%.2f\t%.2f\t%.2f\t%.2f\t%.2f\t%.2f\t\n",
                students[i].studentID, students[i].name, students[i].sex,
                students[i].chinese, students[i].maths, students[i].chemistry,
                students[i].physics, students[i].English, students[i].sumScore);
            found = 1;
        }
    }
    if (!found)
    {
        printf("未找到姓名为 %s 的学生。\n", name);
    }
}
// 按照学号删除某个学生,并输出删除后的所有学生信息
void deleteStudentByID(STUDENT students[], int* count, long studentID)
{
    int index = -1;
    for (int i = 0; i < *count; i++)
    {
        if (students[i].studentID == studentID)
        {
            index = i;
            break;
        }
    }
    if (index != -1)
    {
        for (int i = index; i < *count - 1; i++)
        {
            students[i] = students[i + 1];
        }
        (*count)--;
        printf("学号为 %ld 的学生已删除。\n", studentID);
    }
    else
    {
        printf("未找到学号为 %ld 的学生。\n", studentID);
    }
}


目录
相关文章
|
2月前
|
C语言 C++
【实战指南】 C/C++ 枚举转字符串实现
本文介绍了在C/C++中实现枚举转字符串的实用技巧,通过宏定义与统一管理枚举名,提升代码调试效率并减少维护错误。
189 37
|
2月前
|
程序员 编译器 C++
【实战指南】C++ lambda表达式使用总结
Lambda表达式是C++11引入的特性,简洁灵活,可作为匿名函数使用,支持捕获变量,提升代码可读性与开发效率。本文详解其基本用法与捕获机制。
107 23
|
6月前
|
监控 Linux C++
【实战指南】4步实现C++插件化编程,轻松实现功能定制与扩展(2)
本文是《4步实现C++插件化编程》的延伸,重点介绍了新增的插件“热拔插”功能。通过`inotify`接口监控指定路径下的文件变动,结合`epoll`实现非阻塞监听,动态加载或卸载插件。核心设计包括`SprDirWatch`工具类封装`inotify`,以及`PluginManager`管理插件生命周期。验证部分展示了插件加载与卸载的日志及模块状态,确保功能稳定可靠。优化过程中解决了动态链接库句柄泄露问题,强调了采纳用户建议的重要性。
203 87
【实战指南】4步实现C++插件化编程,轻松实现功能定制与扩展(2)
|
6月前
|
人工智能 程序员 C++
【实战经验】C/C++右移高位补0还是1?
本文探讨了C/C++中右移运算时高位补0还是补1的问题。通过示例代码分析,揭示了右移规则:无符号类型高位补0;有符号类型根据正负决定(正数补0,负数补1)。文中列举了可能导致错误的场景,并提供了两种规避措施——使用无符号类型和掩码校正,确保结果符合预期。最后总结指出,右移运算虽常见,但若处理不当易引发隐晦Bug,需谨慎对待。
328 83
|
7月前
|
存储 监控 算法
员工屏幕监控系统之 C++ 图像差分算法
在现代企业管理中,员工屏幕监控系统至关重要。本文探讨了其中常用的图像差分算法,该算法通过比较相邻两帧图像的像素差异,检测屏幕内容变化,如应用程序切换等。文中提供了C++实现代码,并介绍了其在实时监控、异常行为检测和数据压缩等方面的应用,展示了其实现简单、效率高的特点。
157 15
|
11月前
|
安全 程序员 编译器
【实战经验】17个C++编程常见错误及其解决方案
想必不少程序员都有类似的经历:辛苦敲完项目代码,内心满是对作品品质的自信,然而当静态扫描工具登场时,却揭示出诸多隐藏的警告问题。为了让自己的编程之路更加顺畅,也为了持续精进技艺,我想借此机会汇总分享那些常被我们无意间忽视却又导致警告的编程小细节,以此作为对未来的自我警示和提升。
1160 96
|
10月前
|
自然语言处理 编译器 Linux
告别头文件,编译效率提升 42%!C++ Modules 实战解析 | 干货推荐
本文中,阿里云智能集团开发工程师李泽政以 Alinux 为操作环境,讲解模块相比传统头文件有哪些优势,并通过若干个例子,学习如何组织一个 C++ 模块工程并使用模块封装第三方库或是改造现有的项目。
712 56
|
7月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
3月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
75 0
|
3月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
154 0