学生管理系统(控制台版本)

问题描述:

设计开发一个基于控制台版本的学生管理系统,该系统支持以下功能:

  1. 打印数据库中的所有信息
  2. 录入信息到数据库
  3. 根据输入关键字,删除数据库中的某个同学的信息
  4. 清空数据库
  5. 根据输入关键字,查询数据库中某个同学的信息
  6. 根据输入关键字,修改数据库中某个同学的信息
  7. 对学生信息进行排序

系统说明:

  1. 本系统暂时不用STL库中的vector或者list,而是自己用结构体实现链表
  2. 本系统暂时不用已经成熟的数据库(sqlite,mysql,sqlsever,oracle等),而是用普通的文本文件来简单模拟数据本地存储
  3. 本系统为控制台界面,适合刚刚学习C和C++的同学,操作界面不太友好(如果要界面比较好看,可以用win32,MFC或者其他界面库)
  4. 本系统中的学生信息目前只有学号,姓名,年龄,性别(如果需要其他的信息可以自行完善)
  5. 本系统中对文件的操作使用的函数是fopen,fscanf,fprintf
  6. 本系统中链表的头结点不保存真正的数据信息,只是为了可以方便的删除第一个元素(单链表结构如图所示)

这里写图片描述

参考代码:

student.h文件

#pragma once

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <io.h>
#include <locale>

#define DATABASE_PATH "D:\\student.txt"

typedef struct student
{
    int nGender;//0表示性别位置,1表示男,2表示女
    int nStudentId;
    int nAge;
    char name[50];//人的名字一般不超过50个字符
    struct student *pNext;//下一个节点指针
}STUDENT;

STUDENT *head;//头指针
STUDENT *last;//指向最后一个元素,方便插入

int i, g_nRecordCount = 0, nAge, nStudentId, nGender;
char name[50];

FILE *stream;

void UpdateDataBase();//把信息写入数据库
void InitHeadNode();//初始化头结点
void LoadDataBase();//加载数据库的数据
void PrintStudentInfo();//打印学生信息
void TackleUserOperation();//处理用户选择功能菜单
void InputInfoToDatabase();//用户选择录入信息
int IsStudentIdExist(int nStudentId);//判断学号是否已经存在,因为学号是不能重复的
void DeleteStudentInfoByStudentId();//根据学号删除该同学信息
void ClearDataBase();//情况数据库中数据
void SearchStudentInfo();//查询学生信息
void UpdateStudentInfo();//修改学生信息
void MySort();//对学生信息进行排序
void MySwap(STUDENT *p1, STUDENT *p2);//交换两个节点的信息
void MySwap(int *p1, int *p2);//交换两个整数
void MySwap(char *p1, char *p2);//交换两个字符串
void SoftByType(int(*CMP)(STUDENT *p1, STUDENT *p2));//根据不同的类型进行排序
int CmpByStudentIdAscend(STUDENT *p1, STUDENT *p2);//按照学号从小到大排序
int CmpByStudentIdDescend(STUDENT *p1, STUDENT *p2);//按照学号从大到小排序
int CmpByAgeAscend(STUDENT *p1, STUDENT *p2);//按照年龄从小到大排序
int CmpByAgeDescend(STUDENT *p1, STUDENT *p2);//按照年龄从大到小排序

main.cpp文件

#define _CRT_SECURE_NO_WARNINGS
#include "student.h"

void UpdateDataBase()//把信息写入数据库
{
    if ((stream = fopen(DATABASE_PATH, "w")) == NULL)//打开文件,然后写入
    {
        printf("打开数据库失败\n");
        return;
    }

    STUDENT *p = head->pNext;
    while (p)
    {
        fprintf(stream, "%s %d %d %d\n", p->name, p->nStudentId, p->nAge, p->nGender);//写入文件
        p = p->pNext;
    }
    fclose(stream);
}

void PrintStudentInfo()//打印内存中的链表数据
{
    printf("\n*******************************************************************\n\n");
    if (g_nRecordCount == 0)
    {
        printf("目前数据库中暂无记录\n");
        goto Exit0;
    }
    printf("目前数据库中有 %d 条记录\n", g_nRecordCount);
    printf("学号     姓名      年龄      性别\n");
    STUDENT *p = head->pNext;
    while (p)
    {
        printf("%-10d%-10s%-10d", p->nStudentId, p->name, p->nAge);
        if (p->nGender == 1)
            printf("%-10s", "男");
        else if (p->nGender == 2)
            printf("%-10s", "女");
        else
            printf("%-10s", "未知");
        printf("\n");
        p = p->pNext;
    }

Exit0:
    printf("\n*******************************************************************\n\n");
}

int IsStudentIdExist(int nStudentId)
{
    STUDENT *p = head->pNext;
    while (p)
    {
        if (p->nStudentId == nStudentId)//已经存在
            return 1;

        p = p->pNext;
    }
    return 0;
}

void InputInfoToDatabase()//输入信息到数据库
{
    STUDENT *s = NULL;
    printf("\n请根据提示输入以下信息\n\n");
    printf("学号: ");
    scanf("%d", &nStudentId);
    printf("姓名: ");
    scanf("%s", name);
    printf("年龄: ");
    scanf("%d", &nAge);
    printf("性别(输入整数,1表示男,2表示女,0表示未知): ");
    scanf("%d", &nGender);
    if (IsStudentIdExist(nStudentId))
    {
        printf("\n该学号在数据库已经存在,不能重复,请重新录入\n\n");
        return;
    }
    s = (STUDENT*)malloc(sizeof(STUDENT));
    s->nGender = nGender;
    s->nAge = nAge;
    s->nStudentId = nStudentId;
    strcpy(s->name, name);
    s->pNext = NULL;
    last->pNext = s;
    last = s;//把该节点更新为最后一个数据节点
    g_nRecordCount++;
    UpdateDataBase();
    printf("\n录入成功\n");
}

void DeleteStudentInfoByStudentId()
{
    printf("\n请输入你要删除的学生的学号: \n\n");
    scanf("%d", &nStudentId);
    if (0 == IsStudentIdExist(nStudentId))
    {
        printf("该学号不存在\n");
        return;
    }
    STUDENT *s = head, *p = head->pNext;
    while (p)
    {
        if (p->nStudentId == nStudentId)//删除该学生节点
        {
            s->pNext = p->pNext;
            free(p);
            g_nRecordCount--;
            goto Exit0;
        }
        s = p;
        p = p->pNext;//移动指针
    }

Exit0:
    UpdateDataBase();
    printf("\n删除成功\n");
}

void ClearDataBase()
{
    STUDENT *s = head, *p = head->pNext;
    while (p)
    {
        s->pNext = p->pNext;
        free(p);
        p = s->pNext;
    }
    g_nRecordCount = 0;
    UpdateDataBase();
    printf("\n数据库清空成功\n");
}

void SearchStudentInfo()
{
    printf("\n请输入你要查询的学生的学号: ");
    scanf("%d", &nStudentId);
    if (0 == IsStudentIdExist(nStudentId))
    {
        printf("该学号不存在\n");
        return;
    }

    printf("学号     姓名      年龄      性别\n");
    STUDENT *p = head->pNext;
    while (p)
    {
        if (p->nStudentId == nStudentId)
        {
            printf("%-10d%-10s%-10d", p->nStudentId, p->name, p->nAge);
            if (p->nGender == 1)
                printf("%-10s", "男");
            else if (p->nGender == 2)
                printf("%-10s", "女");
            else
                printf("%-10s", "未知");
            printf("\n");
            break;
        }
        p = p->pNext;
    }
}

void UpdateStudentInfo()
{
    printf("\n请输入你要修改信息的学生的学号: ");
    scanf("%d", &nStudentId);
    if (0 == IsStudentIdExist(nStudentId))
    {
        printf("该学号不存在\n");
        return;
    }

    STUDENT *p = head->pNext;
    while (p)
    {
        if (p->nStudentId == nStudentId)
            break;
        p = p->pNext;
    }

    printf("\n\n请选择你要修改的信息类别\n");
    printf("输入1----修改姓名\n");
    printf("输入2----修改年龄\n");
    printf("输入3----修改性别\n");
    printf("输入其他----不修改\n");

    int nInput;
    scanf("%d", &nInput);
    if (nInput == 1)
    {
        printf("输入修改后的名字: ");
        scanf("%s", name);
        strcpy(p->name, name);
        printf("\n修改成功\n\n");
        UpdateDataBase();
    }
    else if (nInput == 2)
    {
        printf("输入修改后的年龄: ");
        scanf("%d", &nAge);
        p->nAge = nAge;
        printf("\n修改成功\n\n");
        UpdateDataBase();
    }
    else if (nInput == 3)
    {
        printf("输入修改后的性别(1表示男,2表示女,0表示未知): ");
        scanf("%d", &nGender);
        p->nGender = nGender;
        printf("\n修改成功\n\n");
        UpdateDataBase();
    }
}

void MySwap(int *p1, int *p2)
{
    if (NULL == p1 || NULL == p2)
        return;

    int temp = *p1;
    *p1 = *p2;
    *p2 = temp;
}

void MySwap(char *p1, char *p2)
{
    if (NULL == p1 || NULL == p2)
        return;

    char temp[50] = { 0 };
    strcpy(temp, p1);
    strcpy(p1, p2);
    strcpy(p2, temp);
}

void MySwap(STUDENT *p1, STUDENT *p2)
{
    MySwap(&p1->nAge, &p2->nAge);
    MySwap(&p1->nStudentId, &p2->nStudentId);
    MySwap(&p1->nGender, &p2->nGender);
    MySwap(p1->name, p2->name);
}

int CmpByStudentIdAscend(STUDENT *p1, STUDENT *p2)
{
    return p1->nStudentId > p2->nStudentId;
}

int CmpByStudentIdDescend(STUDENT *p1, STUDENT *p2)
{
    return p1->nStudentId < p2->nStudentId;
}

int CmpByAgeAscend(STUDENT *p1, STUDENT *p2)
{
    return p1->nAge > p2->nAge;
}

int CmpByAgeDescend(STUDENT *p1, STUDENT *p2)
{
    return p1->nAge < p2->nAge;
}

void SoftByType(int(*CMP)(STUDENT *p1, STUDENT *p2))
{
    STUDENT *p = NULL;
    STUDENT *q = NULL;
    for (p = head->pNext; p->pNext != NULL; p = p->pNext)
    {
        for (q = p->pNext; q != NULL; q = q->pNext)
        {
            if (CMP(p, q))
            {
                MySwap(p, q);
            }
        }
    }
    printf("\n排序成功\n\n");
    UpdateDataBase();
}

void MySort()
{
    printf("\n\n请选择排序方式\n");
    printf("输入1----按学号从小到大排序\n");
    printf("输入2----按学号从大到小排序\n");
    printf("输入3----按年龄从小到大排序\n");
    printf("输入4----按年龄从大到小排序\n");
    printf("输入其他----不排序\n");
    int nInput;
    scanf("%d", &nInput);
    if (nInput == 1)
        SoftByType(CmpByStudentIdAscend);
    else if (nInput == 2)
        SoftByType(CmpByStudentIdDescend);
    else if (nInput == 3)
        SoftByType(CmpByAgeAscend);
    else if (nInput == 4)
        SoftByType(CmpByAgeDescend);
}

void TackleUserOperation()
{
    while (1)
    {
        printf("\n*************输入数字1-7,其他数字退出系统*******************\n\n");
        printf("输入1----打印数据库中的所有信息\n");
        printf("输入2----录入信息到数据库\n");
        printf("输入3----根据输入关键字,删除数据库中的某个同学的信息\n");
        printf("输入4----清空数据库\n");
        printf("输入5----根据输入关键字,查询数据库中某个同学的信息\n");
        printf("输入6----根据输入关键字,修改数据库中某个同学的信息\n");
        printf("输入7----对学生信息进行排序\n");
        printf("输入其他数字----退出系统\n");
        printf("\n************************************************************\n\n");

        int nOpertion;
        scanf("%d", &nOpertion);
        if (nOpertion == 1)
            PrintStudentInfo();
        else if (nOpertion == 2)
            InputInfoToDatabase();
        else if (nOpertion == 3)
            DeleteStudentInfoByStudentId();
        else if (nOpertion == 4)
            ClearDataBase();
        else if (nOpertion == 5)
            SearchStudentInfo();
        else if (nOpertion == 6)
            UpdateStudentInfo();
        else if (nOpertion == 7)
            MySort();
        else
        {
            printf("已经退出系统\n");
            break;
        }
    }
}

void InitHeadNode()//头部指针存在的目的,是为了可以方便的删除第一个元素
{//头指针的数据初始化为0
    head = (STUDENT*)malloc(sizeof(STUDENT));
    head->nGender = 0;
    strcpy(head->name, "");
    head->nAge = 0;
    head->nStudentId = 0;
    head->pNext = NULL;
    last = head;//还没有数据的时候头尾指针指向同一个位置
}

void LoadDataBase()//加载数据库中的数据
{
    if (_access(DATABASE_PATH, 0))
    {
        printf("数据库文件还不存在,录入信息会自动创建\n");
        return;
    }

    if ((stream = fopen(DATABASE_PATH, "r")) == NULL)//打开文件,然后读取
    {
        printf("打开数据库失败\n");
        return;
    }

    STUDENT *p = head, *s;
    while (fscanf(stream, "%s%d%d%d", name, &nStudentId, &nAge, &nGender) == 4)
    {
        s = (STUDENT*)malloc(sizeof(STUDENT));
        s->nGender = nGender;
        s->nAge = nAge;
        s->nStudentId = nStudentId;
        strcpy(s->name, name);
        s->pNext = NULL;
        last = s;//把该节点更新为最后一个数据节点
        p->pNext = s;
        p = s;
        g_nRecordCount++;
    }

    if (0 == g_nRecordCount)
        printf("目前数据库中暂无记录\n");
    else
        PrintStudentInfo();
    fclose(stream);
}

int main(void)
{
    printf("***********************欢迎进入学生管理系统***********************\n");
    InitHeadNode();
    LoadDataBase();
    TackleUserOperation();
    return 0;
}

运行结果:

这里写图片描述

猜你喜欢

转载自blog.csdn.net/yi_ming_he/article/details/72851666