一、程序要求
该程序是由C语言编写的一个宿舍管理查询软件,其主要功能是实现对学生信息的增加、删除、修改、查询、排序、存储、加载。其中查询可以分别以姓名、学号、寝室号为关键字查询学生信息。排序可以分别以学号和寝室号为关键字进行从小到大的排序。方便舍管阿姨管理宿舍。
二、程序设计
2.1、总体设计:
①输入的形式:根据用户所选择的功能,然后再依据程序相应的提示进行输入:
选择功能1(添加学生信息):按照从先到后的顺序输入学生姓名(30个字符以内),学生学号(15个字符以内),宿舍号码(整数,范围在0~32767)。
选择功能2(修改学生信息):按照从先到后的顺序输入要修改的学生姓名(30个字符以内),修改后学生的姓名(30个字符以内),修改后学生的学号(15个字符以内),修改后学生的宿舍号(整数,范围在0~32767)。
选择功能3(删除学生信息):输入要删除的学生姓名(30个字符以内)。
选择功能4(以姓名查询学生信息):输入要查找的学生的姓名(30个字符以内)。
选择功能5(以学号查询学生信息):输入要查找的学生的学号(15个字符以内)。
选择功能6(以寝室号查询学生信息):输入寝室号(整数,范围在0~32767)。
②输出的形式:根据用户所选择的功能不同,会有不同的输出:
选择功能1(添加学生信息):根据程序的判断会输出添加成功或者添加失败。
选择功能2(修改学生信息):根据程序的判断会输出修改成功或者修改失败。
选择功能3(删除学生信息):根据程序的判断会输出删除成功或者删除失败。
选择功能4(以姓名查询学生信息):根据程序的判断,如果查找成功,则会输出相应的学生信息,反之,则会输出查找失败。
选择功能5(以学号查询学生信息):根据程序的判断,如果查找成功,则会输出相应的学生信息,反之,则会输出查找失败。
选择功能6(以寝室号查询学生信息):根据程序的判断,如果查找成功,则会输出相应的学生信息,反之,则会输出查找失败。
选择功能7,8(以寝室号和学号为关键字进行从小到大的排序):程序会输出排序成功。
选择功能9(显示所有学生的信息):程序会输出所有以存学生的信息。
选择功能10(保存操作):根据程序的判断会输出保存成功或者保存失败。
选择功能11(加载记录):根据程序的判断会输出加载记录成功或者加载记录失败。
2.2、详细设计
2.本程序所包含的15个函数为:
(1).主函数main();
(2).菜单函数Menu();
(3).判断学号是否重复函数S_number_Judge(Student S,int t);
(4).添加学生信息函数Add(Student S);
(5).修改学生信息函数Alter(Student S);
(6).删除学生信息函数Delete(Student S);
(7).显示所有学生信息函数Display_All(Student S);
(8).按照寝室号从小到大排序函数Sort_D_number(Student S);
(9).按照学号从小到大排序函数Sort_S_number(Student S);
(10).根据学生姓名查找函数Query_S_name(Student S);
(11).根据学生学号查找函数Query_S_number(Student S);
(12).根据寝室号查找函数Query_D_number(Student S)
(13).存储函数Save(Student S);
(14).加载函数Load(Student S);
(15).判断在退出程序时是否保存函数Judge_Save(int i,Student S);
各函数之间的调用关系如下:
三、系统实现
部分程序代码:
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#define M 100
int dirty=0; //用来判断是否已保存操作
//定义一个存储学生相关信息的结构体
typedef struct
{
char S_name[31]; //学生姓名
char S_class[31]; //学生班级
char S_number[16]; //学生学号
int D_number; //学生所在寝室的宿舍号
char S_address[500]; //学生地址
int S_phone[20]; //学生电话号码
int Total; //学生总数
}Student[M],St;
//判断学号是否与表中所存学号重复
void S_number_Judge(Student S,int t)
{
int i;
for(i=1;i<=(S->Total)-1;i++)
while(strcmp(S[i].S_number,S[t].S_number)==0)
{
printf("学号输入失败,该学号已存在,请重新输入学号!n");
printf("请输入学生的学号(15个字符以内):");
scanf("%s",S[t].S_number);
getchar();
i=1;
}
}
//添加学生信息函数
void Add(Student S)
{
printf("请输入学生姓名(30个字符以内):");
scanf("%s",S[++(S->Total)].S_name);
getchar();//获取换行符
printf("请输入学生班级(30个字符以内):");
scanf("%s",S[S->Total].S_class);
getchar();
printf("请输入学生的学号(15个字符以内):");
scanf("%s",S[S->Total].S_number);
getchar();
S_number_Judge(S,S->Total);//判断输入的学号是否与表中所存在的学号重复
printf("请输入宿舍号码:");
scanf("%d",&S[S->Total].D_number);
getchar();
printf("请输入学生地址:");
scanf("%s",S[S->Total].S_address);
getchar();
printf("请输入学生电话号码:");
scanf("%s",&S[S->Total]. S_phone);
getchar();
dirty=1;
printf("添加成功!nn");
}
//删除学生信息
void Delete(Student S)
{
int i,j;
int flag=0; //用来判断表中是否存在所要删除的学生的信息
char name[20];
printf("请输入你要删除学生的姓名:");
scanf("%s",name);
getchar();
for(i=1;i<=S->Total;i++)
if(strcmp(S[i].S_name,name)==0)
flag=i;
if(!flag)
printf("你所要删除的学生在表中不存在!");
else
{
for(i=flag;i<S->Total;i++)
{
j=i+1;
strcpy(S[i].S_name,S[j].S_name);
strcpy(S[i].S_number,S[j].S_number);
S[i].D_number=S[j].D_number;
}
(S->Total)--;
dirty=1;
printf("删除成功!");
}
printf("nn");
}
//排序函数按照寝室号从小到大排序(冒泡法)
void Sort_D_number(Student S)
{
int i,j,t;
char name[30];
char number[15];
char address[300];
char Cnumber[30];
for(i=1;i<=S->Total;i++)
for(j=i;j<=S->Total;j++)
if(S[i].D_number>S[j].D_number)
{
strcpy(name,S[i].S_name);
strcpy(number,S[i].S_number);
strcpy(address,S[i].S_address);
strcpy(Cnumber,S[i].S_class);
t=S[i].D_number;
strcpy(S[i].S_name,S[j].S_name);
strcpy(S[i].S_number,S[j].S_number);
strcpy(S[i].S_address,S[j].S_address);;
strcpy(S[i].S_class,S[j].S_class);
S[i].D_number=S[j].D_number;
strcpy(S[j].S_name,name);
strcpy(S[j].S_number,number);
strcpy(S[j].S_address,address);
strcpy(S[j].S_class,Cnumber);
S[j].D_number=t;
}
}
//排序函数按照学号从小到大排序(冒泡法)
//排序函数按照班级号从小到大排序(冒泡法)
void Sort_S_class (Student S)
{
int i,j,t;
char name[30];
char number[15];
char address[300];
char Cnumber[30];
for(i=1;i<=S->Total;i++)
for(j=i;j<=S->Total;j++)
if(strcmp(S[i].S_class,S[j].S_class)>0)
{
strcpy(name,S[i].S_name);
strcpy(number,S[i].S_number);
strcpy(address,S[i].S_address);
strcpy(Cnumber,S[i].S_class);
t=S[i]. D_number;
strcpy(S[i].S_name,S[j].S_name);
strcpy(S[i].S_number,S[j].S_number);
strcpy(S[i].S_address,S[j].S_address);;
strcpy(S[i].S_class,S[j].S_class);
S[i]. D_number =S[j]. D_number;
strcpy(S[j].S_name,name);
strcpy(S[j].S_number,number);
strcpy(S[j].S_address,address);
strcpy(S[j].S_class,Cnumber);
S[j]. D_number =t;
}
}
//查询函数以班级为关键字进行查询(顺序查找)
void Query_S_class(Student S)
{
int i,j=0;
char classnumber[31];
printf("请输入你要查找的班级号(30个字符以内):");
scanf("%s",classnumber);
getchar();
printf("所查找学生信息如下:n");
printf("学生姓名 学生班级 学生学号 宿舍号 学生地址 学生电话号码n");
for(i=1;i<=S->Total;i++)
if(strcmp(classnumber,S[i].S_class)==0)
{
printf("%-20s%-20s %-15s%-5d%s-10%sn",S[i].S_name, S[i].S_class,S[i].S_number,S[i].D_number,S[i].S_address,S[i].S_phone);
j=1;
}
if(!j)
printf("n查找失败,表中不存在该学生的信息!nn");
}
//查询函数以姓名为关键字进行查询(顺序查找)
void Query_S_name(Student S)
{
int i,j=0;
char name[31];
printf("请输入你要查找的学生的姓名(30个字符以内):");
scanf("%s",name);
getchar();
printf("所查找学生信息如下:n");
printf("学生姓名 学生班级 学生学号 宿舍号 学生地址 学生电话号码n");
for(i=1;i<=S->Total;i++)
if(strcmp(name,S[i].S_name)==0)
{
printf("%-20s%-20s %-15s%-5d%s-10%sn",S[i].S_name, S[i].S_class,S[i].S_number,S[i].D_number,S[i].S_address,S[i].S_phone);
j=1;
}
if(!j)
printf("n查找失败,表中不存在该学生的信息!nn");
}
//查询函数以学号为关键字进行查询(折半查找)
void Query_S_number(Student S)
{
int i,j,top,base,mid;
char number[15];
j=0;
base=1;
top=S->Total;
printf("请输入你要查找学生的学号:");
scanf("%s",number);
getchar();
Sort_S_number(S); //将表中原数据按照学号从小到大排序
printf("所查找学生信息如下:n");
printf("学生姓名 学生班级 学生学号 宿舍号 学生地址 学生电话号码n");
if(strcmp(number,S[1].S_number)>=0&&strcmp(number,S[S->Total].S_number)<=0)
{
while(base<=top)
{
mid=(base+top)/2;
if(strcmp(number,S[mid].S_number)==0)
{
printf("%-20s%-20s%-15s%-5d%s-10%sn",S[i].S_name, S[i].S_class,S[i].S_number,S[i].D_number,S[i].S_address,S[i].S_phone);
putchar('n');
j=1;
break;
}
else if(strcmp(number,S[mid].S_number)>0)
base=mid+1;
else
top=mid-1;
}
}
if(!j)
printf("n查找失败,表中不存在该学生的信息!nn");
}
//查询函数以寝室号为关键字进行查询(折半查找)
void Query_D_number(Student S)
{
int i,j,m,n,base,top,mid;
j=0;
base=1;
top=S->Total;
printf("请输入你要查询的寝室号:");
scanf("%d",&i);
getchar();
Sort_D_number(S);//将表中原数据按照寝室号从小到大排序
printf("所查找寝室信息如下:n");
printf("学生姓名 学生班级 学生学号 宿舍号 学生地址 学生电话号码n");
if(i>=S[1].D_number&&i<=S[S->Total].D_number)
{
while(base<=top)
{
mid=(base+top)/2;
if(i==S[mid].D_number)
{
m=mid;
n=mid-1;
while(S[m].D_number==i)
{
printf("%-20s%-20s %-15s%-5d%s-10%sn",S[m].S_name, S[i].S_class,S[m].S_number,S[m].D_number,S[i].S_address,S[i].S_phone);
m++;
if(m>S->Total)
break;
}
if(n>0)
{
while(S[n].D_number==i)
{
printf("%-20s%-15s%-5dn",S[n].S_name,S[n].S_number,S[n].D_number);
n--;
if(n<1)
break;
}
}
j=1;
putchar('n');
break;
}
else if(i>S[mid].D_number)
base=mid+1; //折半查找
else
top=mid-1;
}
}
if(!j)
printf("n查找失败,表中不存在该寝室的信息!nn");
}
//存储函数
void Save(Student S)
{
St Std;
int i;
int flag1=0,flag2=0; //判断存储是否成功
FILE *fp;
if((fp=fopen("宿舍管理.txt","w"))==NULL)
{
printf("打开文件失败!nn");
flag1=1;
exit(0); //结束程序
}
for(i=1;i<=S->Total;i++)
if(fwrite(&S[i],sizeof(Std),1,fp)!=1)
{
printf("数据写入错误nn");
flag2=1;
exit(0);
}
if(!flag1&&!flag2)
{
printf("数据存储成功!nn");
dirty=0;
}
fclose(fp); //关闭一个流。关闭一个打开的文件, fp是此前通过调用fopen返回的文件指针。
}
//加载记录函数
void Load(Student S)
{
St Std;
FILE *fp;
if((fp=fopen("宿舍管理.txt","r"))==NULL)
{
printf("打开文件失败!nn");
exit(0);
}
while(!feof(fp)) // fp为文件句柄,feof为判断是否读到文件尾
fread(&S[++(S->Total)],sizeof(Std),1,fp);
fclose(fp);
printf("加载数据成功!nn");
(S->Total)--; //由于读取问题,表中个数要减去
}
//退出程序时判断是否保存函数
void Judge_Save(int i,Student S)
{
char ch;
if(i)
{
printf("表中数据已改变,是否保存后再退出(Y/N)?:");
ch=getchar();
getchar();
while(ch!='n'&&ch!='N'&&ch!='y'&&ch!='Y')
{
printf("请输入N(n)或者Y(y):");
ch=getchar();
getchar();
}
if(ch=='y'||ch=='Y')
Save(S);
}
}
//菜单
void Menu()
{
//菜单
printf(" 宿舍管理nn");
printf("*************************************菜单*************************************n");
printf("1.添加新入住学生信息 | 8.按照班级号从小到大排序n");
printf("2.修改宿舍学生信息 | 9.按照寝室号从小到大排序n");
printf("3.删除搬出宿舍学生信息 | 10.按照学号从小到大排序n");
printf("4.以姓名查询学生信息 | 11.显示所有学生的相关信息n");
printf("5.以学号查询学生信息 | 12.保存操作n");
printf("6.以宿舍号查询该宿舍中的全部学生信息 | 13.加载记录n");
printf("7.以班级号查询该班级中的全部学生信息 | 14.退出程序n");
printf("******************************************************************************n");
}
void main()
{
int i;
Student S;
S->Total=0;
do
{
Menu();
printf("请选择所要实现的功能(请输入1~14中的任意一个数字):");
scanf("%d",&i);
getchar(); //获取换行符
putchar('n');
switch(i)
{
case 1:Add(S);
break;
case 2:Alter(S);
break;
case 3:Delete(S);
break;
case 4:Query_S_name(S);
break;
case 5:Query_S_number(S);
break;
case 6:Query_D_number(S);
break;
case 7: Query_S_class (S);
break;
case 8: Sort_S_class (S);
printf("排序完成!nn");
break;
case 9:Sort_D_number(S);
printf("排序完成!nn");
break;
case 10:Sort_S_number(S);
printf("排序完成!nn");
break;
case 11:Display_All(S);
break;
case 12:Save(S);
break;
case 13:Load(S);
break;
case 14:Judge_Save(dirty,S);
exit(0);
break;
default:printf("选择错误:请在选项到之间选择!nn");
break;
}
}while(i!=14);
}
四、系统调试
1、初次使用在菜单中选1输入入住学生信息
按照先后顺序分别输入:
姓名:吴武,学号:20132255,班级:高三1班,宿舍号:404 ,地址27#,号码13878797218
姓名:周王,学号:20132551,班级:高三1班,宿舍号:402,地址:27#,号码 15296812933
姓名:秦学,学号:20135599,班级:高三2班,宿舍号:403,地址:27#,号码 15240662499
分别如下图:
2、对功能2(修改学生信息)的测试:修改学生秦学的信息
如图
3、对功能3(删除学生信息)的测试:删除搬出宿舍学生吴武的信息
如图
4、对功能4(以姓名查询学生信息)的测试:查询学生秦学的信息
如图
5、对功能6的测试:查询宿舍402
6、对功能9进行测试:
需要全部资料可在我的资源里下载,内容包括程序文档、源程序。
需要完整的资料可在我的资源里下载,也可以加入我的纷传圈子,里面有资源压缩包的百度网盘下载地址及提取码。
纷传点击用微信打开即可,过程有点繁琐请见谅。