本篇博客介绍一个关于学生成绩管理系统的C语言代码,包含读取成绩、计算各门课程的总分和平均分、按分数排序、按学号排序、按姓名排序、查找成绩等基本功能。
一、实现思路
1、定义STU结构体,存储学生的基本信息和成绩。
2、编写各个函数实现不同的功能,包括输入学生成绩、计算总分和平均分、排序、查找、统计分数段、输出到文件等。
3、在主函数中编写程序逻辑,按用户选择调用不同的函数,实现学生成绩管理系统的各项功能。
4、运行程序,查看输出结果或将结果存储到文件中,实现数据的长期保存和管理。
二、代码的实现
在开始之前需要创建三个文件:
test.c — 用来构建学生成绩管理系统的整体逻辑
student.h — 用来存放函数所需要的头文件和声明
student.c — 存放函数的主体
然后就可以来完成学生成绩管理系统的代码了
(1)构造功能框架
此步是将系统主要功能的算法框架建立出来(只是框架,不需要实现功能)
代码如下
int main()
{
char ch;
int n = 0, m = 0;
STU stu[STU_NUM];
printf("Input student number(n<%d):", STU_NUM);
scanf("%d", &n);
printf("Input course number(m<=%d):", COURSE_NUM);
scanf("%d", &m);
while (1)
{
ch = Menu(); // 显示菜单,并读取用户输入
switch (ch)
{
case 1:ReadScore(stu, n, m);
break;
case 2: AverSumofEveryCourse(stu, n, m);
break;
case 3: AverSumofEveryStudent(stu, n, m);
break;
case 4: SortbyScore(stu, n, m, Descending);
printf("\nSort in descending order by score:\n");
PrintScore(stu, n, m);
break;
case 5: SortbyScore(stu, n, m, Ascending);
printf("\nSort in ascending order by score:\n");
PrintScore(stu, n, m);
break;
case 6: AsSortbyNum(stu, n, m);
printf("\nSort in ascending order by number:\n");
PrintScore(stu, n, m);
break;
case 7: SortbyName(stu, n, m);
printf("\nSort in dictionary order by name:\n");
PrintScore(stu, n, m);
break;
case 8: SearchbyNum(stu, n, m);
break;
case 9: SearchbyName(stu, n, m);
break;
case 10: StatisticAnalysis(stu, n, m);
break;
case 11:PrintScore(stu, n, m);
break;
case 12:WritetoFile(stu, n, m);
break;
case 13:ReadfromFile(stu, &n, &m);
break;
case 0: printf("End of program!");
exit(0);
default:printf("Input error!");
}
}
return 0;
}
(2)实现各项功能
接下来,我们需要实现功能函数,如读取成绩、计算每个学生各门课程的总分和平均分、计算每门课程的总分和平均分等:
// 函数功能:输入n个学生的m门课成绩
void ReadScore(STU stu[], int n, int m)
{
int i, j;
printf("Input student's ID, name and score:\n");
for (i = 0; i < n; i++)
{
printf("%d:", i + 1);
scanf("%ld %s", &stu[i].num, stu[i].name);
for (j = 0; j < m; j++)
{
scanf("%f", &stu[i].score[j]);
}
}
}
// 函数功能:计算每个学生各门课程的总分和平均分
void AverSumofEveryStudent(STU stu[], int n, int m)
{
int i, j;
for (i = 0; i < n; i++)
{
stu[i].sum = 0;
for (j = 0; j < m; j++)
{
stu[i].sum = stu[i].sum + stu[i].score[j];
}
stu[i].aver = m > 0 ? stu[i].sum / m : -1;
printf("student %d: sum = %.0f, aver = %.0f\n", i + 1, stu[i].sum, stu[i].aver);
}
}
// 函数功能:计算每门课程的总分和平均分
void AverSumofEveryCourse(STU stu[], int n, int m)
{
int i, j;
float sum[COURSE_NUM], aver[COURSE_NUM];
memset(sum, 0, sizeof(sum)); // 将数组sum的全部元素初始化为0
memset(aver, 0, sizeof(aver)); // 将数组aver的全部元素初始化为0
for (i = 0; i < n; i++)
{
for (j = 0; j < m; j++)
{
sum[j] += stu[i].score[j];
}
}
for (j = 0; j < m; j++)
{
aver[j] = n > 0 ? sum[j] / n : -1;
printf("Course %d: sum=%.0f, aver=%.2f\n", j + 1, sum[j], aver[j]);
}
}
接着,我们需要实现按分数排序、按学号排序、按姓名排序的功能,这里使用选择法排序和交换法排序,代码如下:
// 函数功能:按选择法将数组sum的元素值排序
void SortbyScore(STU stu[], int n, int m, int (*compare)(float a, float b))
{
int i, j, k, t;
for (i = 0; i < n - 1; i++)
{
k = i;
for (j = i + 1; j < n; j++)
{
if ((*compare)(stu[j].sum, stu[k].sum)) k = j;
}
if (k != i)
{
for (t = 0; t < m; t++) // 交换m门课程的成绩
{
SwapFloat(&stu[k].score[t], &stu[i].score[t]);
}
SwapFloat(&stu[k].sum, &stu[i].sum); // 交换总分
SwapFloat(&stu[k].aver, &stu[i].aver); // 交换平均分
SwapLong(&stu[k].num, &stu[i].num); // 交换学号
SwapChar(stu[k].name, stu[i].name); // 交换姓名
}
}
}
// 使数据按升序排序
int Ascending(float a, float b)
{
return a < b; // 这样比较决定了按升序排序,如果a<b,则交换
}
// 使数据按降序排序
int Descending(float a, float b)
{
return a > b; // 这样比较决定了按降序排序,如果a>b,则交换
}
// 交换两个单精度浮点型数据
void SwapFloat(float* x, float* y)
{
float temp;
temp = *x;
*x = *y;
*y = temp;
}
// 交换两个长整型数据
void SwapLong(long* x, long* y)
{
long temp;
temp = *x;
*x = *y;
*y = temp;
}
// 交换两个字符串
void SwapChar(char x[], char y[])
{
char temp[MAX_LEN];
strcpy(temp, x);
strcpy(x, y);
strcpy(y, temp);
}
// 函数功能:按选择法将数组num的元素值按从低到高排序
void AsSortbyNum(STU stu[], int n