问题描述:
设计开发一个基于控制台版本的学生管理系统,该系统支持以下功能:
- 打印数据库中的所有信息
- 录入信息到数据库
- 根据输入关键字,删除数据库中的某个同学的信息
- 清空数据库
- 根据输入关键字,查询数据库中某个同学的信息
- 根据输入关键字,修改数据库中某个同学的信息
- 对学生信息进行排序
系统说明:
- 本系统暂时不用STL库中的vector或者list,而是自己用结构体实现链表
- 本系统暂时不用已经成熟的数据库(sqlite,mysql,sqlsever,oracle等),而是用普通的文本文件来简单模拟数据本地存储
- 本系统为控制台界面,适合刚刚学习C和C++的同学,操作界面不太友好(如果要界面比较好看,可以用win32,MFC或者其他界面库)
- 本系统中的学生信息目前只有学号,姓名,年龄,性别(如果需要其他的信息可以自行完善)
- 本系统中对文件的操作使用的函数是fopen,fscanf,fprintf
- 本系统中链表的头结点不保存真正的数据信息,只是为了可以方便的删除第一个元素(单链表结构如图所示)
参考代码:
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;
}
运行结果: