C语言程序设计——设计一个学生管理系统(完美运行的程序(●‘◡‘●))

简介: C语言程序设计——设计一个学生管理系统(完美运行的程序(●‘◡‘●))

一、设计目的


通过c语言设计一个学生管理系统,要求有直观的主菜单,可以录入学生的信息,实现添加学生信息、显示学生信息、查找学生信息、删除学生信息、修改学生信息以及退出等功能。


二、原理及相关功能


(一)基本框架


1、首先因为学生有以下几个基本信息:姓名、年龄、学号、性别,依次由name、age、id和sex来表示,所以可以通过结构体实现,struct语句定义存储不同类型的数据项,定义一个结构体名为student,用于存储每个学生的信息,另外定义一个结构体名为class_room班级,它包含了结构体student的变量初始化st,用于存储学生以及当前班级人数,且定义学生最大人数为60,如下代码:

#define MAX 60//定义MAX最大值为60 
struct student{
  char name[20];
  int age;
  int id;
  char sex[10];
};
struct class_room{
  struct student st[MAX]; //定义多个学生 
  int n;      //当前班级的人数
};


2、由于是显示一个管理系统,所以我们设计一个主菜单,通过定义一个函数printf_menu()来实现这个功能,如下代码:

//打印主菜单函数 
void printf_menu()//打印主菜单函数 
{
  printf("        学生管理系统        \n");
  printf("----------------------------\n");
  printf("|1、添加学生信息            |\n");
  printf("|2、显示所有学生信息        |\n");
  printf("|3、查询学生信息            |\n");
  printf("|4、删除学生信息            |\n");
  printf("|5、修改学生信息            |\n");
  printf("|6、退出                    |\n");
  printf("----------------------------\n");
  printf("请输入相应的序号选择!       \n");
}

3、因为要通过输入相应的序号来选择相应的功能,所以可以在主函数中通过一个switch()语句来实现,若要使程序一直执行下去,通过用户输入退出才退出程序,即设置一个while(1)无限循环下去,另外还要设置一个loop语句,通过goto语句,即若输入错误的序号即跳到选择序号的页面。


访问结构的成员,通过使用运算符.来实现,即WLW.n=0,表示初始化班级WLW的成员n,而&WLW表示取WLW的地址,取出其对应存储空间的值,即存储的学生,另外若想使用指向该结构的指针来访问结构体,通过操作符->实现。

如以下代码:

//主函数
int main() 
{
  struct class_room WLW;//定义一个班级为WLW存储学生 
  WLW.n=0;//初始化,学生人数为0 
  while(1)//无限循环
  {
  loop:
    printf_menu();//调用主菜单函数输出主菜单
    int choose;//定义一个序号
    scanf("%d",&choose);
  switch(choose)
  {
    case 1:
    add_student(&WLW);//添加学生
    break;
    case 2:
    show_student(&WLW);//显示学生
    break;
    case 3:
    find_student(&WLW);//查询学生
    break;
    case 4:
    remove_student(&WLW);//删除学生
    break; 
    case 5:
    change_student(&WLW);//修改学生
    break;
    case 6:
    return 0;//退出程序
    default://若输出错误的序号,则跳转至重新输出
    printf("输出错误,请重新输入!\n");
    goto loop;  
  }
  }
}


(二)功能实现


1、定义一个add_student()函数添加学生信息,参数为struct class_room *WLW,即结构体指针变量。姓名和性别由于是字符串类型,所以scanf()中通过%s格式符表示,年龄和id它们的地址是班级WLW当前学生的相应信息,由于数组名代表首地址这里不用&,另外每次添加后,当前班级的人数n++,如下代码:

//添加学生信息
void add_student(struct class_room *WLW)
{
  printf("请输入学生的姓名:\n");
  scanf("%s",WLW->st[WLW->n].name); //数组名代表首地址 
  printf("请输入学生的年龄:\n");
  scanf("%d",&WLW->st[WLW->n].age); //取变量的地址 
  printf("请输入学生的id:\n");
  scanf("%d",&WLW->st[WLW->n].id);
  printf("请输入学生的性别:\n");
  scanf("%s",WLW->st[WLW->n].sex);
  WLW->n++;//班级人数加一
}


2、定义一个show_student()函数显示学生信息,即显示当前班级的所有学生信息,其中WLW->n为当前班级的人数,如下代码:

//显示学生信息 
void show_student(struct class_room *WLW)
{
  int i;
  for(i=0;i<WLW->n;i++)//WLW->n为当前班级的人数
  {
  printf("the %d student name is %s\n",i+1,WLW->st[i].name);
  printf("the %d student age is %d\n",i+1,WLW->st[i].age);
  printf("the %d student id is %d\n",i+1,WLW->st[i].id);
  printf("the %d student sex is %s\n",i+1,WLW->st[i].sex);
  }
}


3、定义一个find_student()函数查找学生信息,通过输入学生id查找学生的信息,若存在则通过循环依次输出该学生的信息:

//查找学生
int find_student(struct class_room *WLW)
{
  int id,i; 
  printf("请输入要查找的学生id:\n");
  scanf("%d",&id);
  for(i=0;i<WLW->n;i++)
  {
  if(id==WLW->st[i].id)
  {
    printf("the student is exist!\n");
    printf("the %d student name is %s\n",i+1,WLW->st[i].name);
    printf("the %d student age is %d\n",i+1,WLW->st[i].age);
    printf("the %d student id is %d\n",i+1,WLW->st[i].id);
    printf("the %d student sex is %s\n",i+1,WLW->st[i].sex);
    return i;
  }
  }
  printf("the student is not exist!\n");
  return -1;
}


4、定义一个remove_student()函数删除学生信息,这里首先定义一个参数ret,并调用find_student(WLW)查找要删除的学生是否在其中,如果返回值为-1,则进行覆盖,即要删除的学生信息被后面的学生信息所覆盖,另外其中由于于是字符串类型复制时通过使用头文件string.h中的strcpy复制函数从而实现覆盖的功能,然后当前班级学生人数减一:

//删除指定学生 
void remove_student(struct class_room *WLW){
  int ret,i;
  ret=find_student(WLW);
  if(ret!=-1)
  {
  for(i=ret;i<WLW->n-1;i++)
  {
    strcpy(WLW->st[i].name,WLW->st[i+1].name);//由于是字符串类型复制使用头文件string.h中的strcpy复制函数 
    WLW->st[i].age=WLW->st[i+1].age;
    WLW->st[i].id=WLW->st[i+1].id;
    strcpy(WLW->st[i].sex,WLW->st[i+1].sex);
  }
  WLW->n--;
  }
  printf("该学生已经删除成功!\n");
}


5、定义一个change_student()函数修改学生信息,因为学生学习有四项,所以这里通过switch语句、goto语句来实现,定义ret参数调用find_student(WLW)查找要删除的学生是否在其中,然后再修改:

//修改学生信息 
void change_student(struct class_room *WLW)
{
  int ret,choose;
  ret=find_student(WLW);
  if(ret!=-1)
  {
  loop1:
    printf("修改学生信息的哪一项?\n");
    printf("1、姓名\n");
    printf("2、年龄\n");
    printf("3、id\n");
    printf("4、性别\n");
    scanf("%d",&choose);
    switch(choose)
    {
    case 1:
      printf("请输入新的学生姓名:\n");
      scanf("%s",WLW->st[ret].name);//输出的代表首地址,所以不需要&取地址 
      break;
    case 2:
      printf("请输入新的学生年龄:\n");
      scanf("%d",&WLW->st[ret].age);
      break;
    case 3:
      printf("请输入新的学生id:\n");
      scanf("%d",&WLW->st[ret].id);
      break;
    case 4: 
      printf("请输入新的学生性别:\n");
      scanf("%s",WLW->st[ret].sex);
      break;
    default:
      printf("输出错误,请重新输入!\n");
      goto loop1;
    }
  }
}


三、完整代码


以下是完整的程序代码:

/*学生管理系统*/
#include<stdio.h>
#include<string.h>//包含头文件string.h
#define MAX 60//定义MAX最大值为60 
struct student{
  char name[20];
  int age;
  int id;
  char sex[10];
};
struct class_room{
  struct student st[MAX]; //定义多个学生 
  int n;  //当前班级的人数
};
void printf_menu()//打印主菜单函数 
{
  printf("        学生管理系统        \n");
  printf("----------------------------\n");
  printf("|1、添加学生信息            |\n");
  printf("|2、显示所有学生信息        |\n");
  printf("|3、查询学生信息            |\n");
  printf("|4、删除学生信息            |\n");
  printf("|5、修改学生信息            |\n");
  printf("|6、退出                    |\n");
  printf("----------------------------\n");
  printf("请输入相应的序号选择!       \n");
}
void add_student(struct class_room *WLW)//添加学生信息,其中struct class_room *WLW为结构体指针 
{
  printf("请输入学生的姓名:\n");
  scanf("%s",WLW->st[WLW->n].name); //数组名代表首地址 
  printf("请输入学生的年龄:\n");
  scanf("%d",&WLW->st[WLW->n].age); //取变量的地址 
  printf("请输入学生的id:\n");
  scanf("%d",&WLW->st[WLW->n].id);
  printf("请输入学生的性别:\n");
  scanf("%s",WLW->st[WLW->n].sex);
  WLW->n++;//班级人数加一
}
void show_student(struct class_room *WLW)//显示所有学生信息 
{
  int i;
  for(i=0;i<WLW->n;i++)//WLW->n为当前班级的人数
  {
  printf("the %d student name is %s\n",i+1,WLW->st[i].name);//第一个学生
  printf("the %d student age is %d\n",i+1,WLW->st[i].age);
  printf("the %d student id is %d\n",i+1,WLW->st[i].id);
  printf("the %d student sex is %s\n",i+1,WLW->st[i].sex);
  }
}
int find_student(struct class_room *WLW)//查找指定学生 
{
  int id,i; 
  printf("请输入要查找的学生id:\n");
  scanf("%d",&id);
  for(i=0;i<WLW->n;i++)
  {
  if(id==WLW->st[i].id)
  {
    printf("the student is exist!\n");
    printf("the %d student name is %s\n",i+1,WLW->st[i].name);
    printf("the %d student age is %d\n",i+1,WLW->st[i].age);
    printf("the %d student id is %d\n",i+1,WLW->st[i].id);
    printf("the %d student sex is %s\n",i+1,WLW->st[i].sex);
    return i;
  }
  }
  printf("the student is not exist!\n");
  return -1;
}
void remove_student(struct class_room *WLW)//删除指定学生 
{
  int ret,i;
  ret=find_student(WLW);
  if(ret!=-1)
  {
  for(i=ret;i<WLW->n-1;i++)
  {
    strcpy(WLW->st[i].name,WLW->st[i+1].name);//由于是字符串类型复制使用头文件string.h中的strcpy复制函数 
    WLW->st[i].age=WLW->st[i+1].age;
    WLW->st[i].id=WLW->st[i+1].id;
    strcpy(WLW->st[i].sex,WLW->st[i+1].sex);
  }
  WLW->n--;
  }
  printf("该学生已经删除成功!\n");
}
void change_student(struct class_room *WLW)//修改学生信息 
{
  int ret,choose;
  ret=find_student(WLW);
  if(ret!=-1)
  {
  loop1:
    printf("修改学生信息的哪一项?\n");
    printf("1、姓名\n");
    printf("2、年龄\n");
    printf("3、id\n");
    printf("4、性别\n");
    scanf("%d",&choose);
    switch(choose)
    {
    case 1:
      printf("请输入新的学生姓名:\n");
      scanf("%s",WLW->st[ret].name);//输出的代表首地址,所以不需要&取地址 
      break;
    case 2:
      printf("请输入新的学生年龄:\n");
      scanf("%d",&WLW->st[ret].age);
      break;
    case 3:
      printf("请输入新的学生id:\n");
      scanf("%d",&WLW->st[ret].id);
      break;
    case 4: 
      printf("请输入新的学生性别:\n");
      scanf("%s",WLW->st[ret].sex);
      break;
    default:
      printf("输出错误,请重新输入!\n");
      goto loop1;
    }
  }
}
int main() 
{
  struct class_room WLW;//定义一个班级为WLW存储学生 
  WLW.n=0;//初始化,学生人数为0 
  while(1)//无限循环 
  {
  loop:
    printf_menu();//调用主菜单函数输出主菜单
    int choose;//定义一个序号
    scanf("%d",&choose);
  switch(choose)
  {
    case 1:
    add_student(&WLW);//添加学生
    break;
    case 2:
    show_student(&WLW);//显示学生
    break;
    case 3:
    find_student(&WLW);//查询学生
    break;
    case 4:
    remove_student(&WLW);//删除学生
    break; 
    case 5:
    change_student(&WLW);//修改学生
    break;
    case 6:
    return 0;//退出程序
    default://若输出错误的序号,则跳转至重新输出
    printf("输出错误,请重新输入!\n");
    goto loop;  
  }
  }
}


四、运行结果


运行结果:

1667131730635.jpg

添加三个学生的信息,第一个学生:

1667131742191.jpg

第二个学生:

1667131751908.jpg

第三个学生:

1667131762203.jpg

显示所有学生信息:

1667131771693.jpg

查询学生信息,例查询id为100的学生:

1667131779922.jpg

查询学生信息,例查询一个不存在的学生,查询id为103的学生:

1667131788666.jpg

删除学生信息,删除id为102和105的学生,其中id为105的学生不存在:

1667131800385.jpg

1667131808495.jpg

修改id为100的学生姓名为yyy:

1667131818733.jpg

修改后的所有学生信息:

1667131827884.jpg

退出程序:

1667131836639.jpg

相关文章
|
11天前
|
监控 关系型数据库 MySQL
【01】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-硬件设备实时监控系统运营版发布-本产品基于企业级开源项目Zabbix深度二开-分步骤实现预计10篇合集-自营版
【01】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-硬件设备实时监控系统运营版发布-本产品基于企业级开源项目Zabbix深度二开-分步骤实现预计10篇合集-自营版
20 0
|
2月前
|
存储 编译器 C语言
【C语言】C语言的变量和声明系统性讲解
在C语言中,声明和定义是两个关键概念,分别用于告知编译器变量或函数的存在(声明)和实际创建及分配内存(定义)。声明可以多次出现,而定义只能有一次。声明通常位于头文件中,定义则在源文件中。通过合理组织头文件和源文件,可以提高代码的模块化和可维护性。示例包括全局变量、局部变量、函数、结构体、联合体、数组、字符串、枚举和指针的声明与定义。
81 12
|
2月前
|
IDE 编译器 开发工具
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
在本文中,我们系统地讲解了常见的 `#pragma` 指令,包括其基本用法、编译器支持情况、示例代码以及与传统方法的对比。`#pragma` 指令是一个强大的工具,可以帮助开发者精细控制编译器的行为,优化代码性能,避免错误,并确保跨平台兼容性。然而,使用这些指令时需要特别注意编译器的支持情况,因为并非所有的 `#pragma` 指令都能在所有编译器中得到支持。
181 41
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
|
3月前
|
机器学习/深度学习 算法 数据挖掘
C语言在机器学习中的应用及其重要性。C语言以其高效性、灵活性和可移植性,适合开发高性能的机器学习算法,尤其在底层算法实现、嵌入式系统和高性能计算中表现突出
本文探讨了C语言在机器学习中的应用及其重要性。C语言以其高效性、灵活性和可移植性,适合开发高性能的机器学习算法,尤其在底层算法实现、嵌入式系统和高性能计算中表现突出。文章还介绍了C语言在知名机器学习库中的作用,以及与Python等语言结合使用的案例,展望了其未来发展的挑战与机遇。
77 1
|
3月前
|
人工智能 安全 算法
基于C语言的嵌入式系统开发,涵盖嵌入式系统概述、C语言的优势、开发流程、关键技术、应用实例及面临的挑战与未来趋势。
本文深入探讨了基于C语言的嵌入式系统开发,涵盖嵌入式系统概述、C语言的优势、开发流程、关键技术、应用实例及面临的挑战与未来趋势。C语言因其高效、可移植、灵活及成熟度高等特点,在嵌入式系统开发中占据重要地位。文章还介绍了从系统需求分析到部署维护的完整开发流程,以及中断处理、内存管理等关键技术,并展望了嵌入式系统在物联网和人工智能领域的未来发展。
150 1
|
3月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
99 1
|
3月前
|
C语言
C语言编程中,错误处理至关重要,能提升程序的健壮性和可靠性
C语言编程中,错误处理至关重要,能提升程序的健壮性和可靠性。本文探讨了C语言中的错误类型(如语法错误、运行时错误)、基本处理方法(如返回值、全局变量、自定义异常处理)、常见策略(如检查返回值、设置标志位、记录错误信息)及错误处理函数(如perror、strerror)。强调了不忽略错误、保持处理一致性及避免过度处理的重要性,并通过文件操作和网络编程实例展示了错误处理的应用。
101 4
|
3月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
99 5
|
3月前
|
网络协议 物联网 数据处理
C语言在网络通信程序实现中的应用,介绍了网络通信的基本概念、C语言的特点及其在网络通信中的优势
本文探讨了C语言在网络通信程序实现中的应用,介绍了网络通信的基本概念、C语言的特点及其在网络通信中的优势。文章详细讲解了使用C语言实现网络通信程序的基本步骤,包括TCP和UDP通信程序的实现,并讨论了关键技术、优化方法及未来发展趋势,旨在帮助读者掌握C语言在网络通信中的应用技巧。
81 2
|
3月前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
72 1

热门文章

最新文章