【用C语言描述数据结构】课程设计:歌手比赛系统(1)

文末附有全部代码哦💖💖
文末附有全部代码哦~

一、具体问题

1.问题描述:

设计一个简单的歌手比赛绩管理程序,对一次歌手比赛的成绩进行管理,使用链表实现,基本的设计要求如下(不限于此要求):
(1)输入每个选手的数据包括编号、姓名、十个评委的成绩,根据输入计算出总成绩和平均成绩(去掉最高分,去掉最低分)。
(2)显示主菜单如下:

  • 1.选手数据输入
  • 2.评委打分
  • 3.成绩排序
  • 4.数据查询
  • 5.追加选手数据
  • 6.删除选手数据
  • 7.修改选手数据
  • 8.输出全部选手信息
  • 9.写入数据文件
  • 10.退出系统

2.问题来源

当下,无论是校园内还是在社会生活中,经常会举行一些歌手演唱比赛,在信息化的时代里,不会再像以前那样用笔和纸记录比赛的赛况,歌手比赛系统应运而生。歌手比赛系统包含了一切歌手演唱比赛需要的功能,歌手信息的输入,评委打分,成绩排序等功能操作简单高效,可以完成很多繁琐的工作,完全可以满足一切歌手比赛的需求。开发此系统只为方便歌手演唱比赛的举办。

3.解决手段

歌手比赛系统是一个可以储存多个比赛选手信息的系统,主要采用链表来实现存储信息的需求,使用头插法来输入数据,通过选手编号来实现评委打分,成绩排序,数据查询,追加选手数据,删除选手信息,修改选手信息,输出全部选手信息,写入数据文件等功能。

4.应用前景

在信息技术迅速发展的今天,网络对于大多数人已经不再陌生,网络在人们的工作、学习和生活中发挥着不可替代的作用,大大提高了人们生产生活效率。歌手比赛系统是信息技术在人们工作生活中应用的代表,该系统完成评委打分,成绩排序,数据查询,追加选手数据,删除选手信息,修改选手信息,输出全部选手信息,写入数据文件等功能,这些功能可以让评委们在歌手比赛结算时时更加方便快捷。同时,该系统也可以应用到其他比赛当中,使比赛更加方便快捷。

二、需求分析

1.功能需求分析

歌手比赛系统是我们为了方便进行歌手比赛而开发的一套系统,该系统的主要目的是实现选手的信息分析,该系统具有评委打分,成绩排序,数据查询,追加选手数据,删除选手信息,修改选手信息,输出全部选手信息,写入数据文件等功能。同时,歌手比赛系统基于c语言编译的源程序,其调用函数根据所需要模块进行引用,再编写好之后,要对程序进行相应的调试,以验证程序的正确性和可用性。调试及测试时,通过相应信息,充分证明程序的可行性,同时本程序具有很高的逻辑性和严密性。

2.系统功能模块图

该程序的系统功能模块图如下图所示。
在这里插入图片描述

3.数据结构定义

typedef struct Node {
    int id;//编号
    char name[MAXSIZE];//姓名
    float grade[NUMBER];//评委评分
    float sum;//总分
float average;//平均分
struct Node \*next;//指针域
} S;//结构体定义为S

//各函数定义
void menu(); //菜单函数 
S \*create();//创建链表函数 
void print(S \*);//输出链表函数 
void insert(S \*);//插入节点函数 
void del(S \*);//删除节点函数
void update(S \*);

void search(S \*);//查找节点函数
void sort(S \*);//节点排序函数 
void save(S \*);

void giveScore(S \*);

三、系统设计

1.主要函数表

在这里插入图片描述

2.功能流程图

(1)添加选手信息

如下图所示:
在这里插入图片描述

(2)修改选手信息

如下图所示:
在这里插入图片描述

(3)删除选手信息

如下图所示:
在这里插入图片描述

四、代码测试

1.主函数

//主函数
int main() {
    S \*head;
    int n, a = 1;//n用来控制选择操作类型,a控制循环,以-1终止
    while (a > 0) {
        menu();//显示菜单
        printf("请输入操作数:");
        scanf("%d", &n);//选择操作
        switch (n)//各操作数字对应菜单数字,通过n确定操作类型
        {
            //创建结点,输入选手数据
            case 1:
                head = create();
                break;
                //评委打分
            case 2:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                giveScore(head);
                break;
                //排序
            case 3:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                sort(head);
                break;
                //查询
            case 4:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                search(head);
                break;
                //追加数据
            case 5:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                insert(head);
                break;
                //删除
            case 6:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                del(head);
                break;
                //修改数据
            case 7:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                update(head);
                break;
                //输出全部选手信息
            case 8:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                print(head);
                break;
                //写入文件
            case 9:
                if (head == NULL) {
                    printf("链表为空,请先执行1操作!\n");
                    break;
                }
                save(head);
                break;
                //退出系统
            case 0:
                exit(0);
                break;
            default:
                printf("退出系统");
                a = -1;//跳出循环条件
                break;
        }
    }
    return 0;
}


2.菜单模块

//菜单模块直接显示 
void menu() {
    printf("==============================\n");
    printf("=========请选择操作============\n");
    printf("========1.输入选手数据=========\n");
    printf("========2.评委打分============\n");
    printf("========3.成绩排序============\n");
    printf("========4.数据查询============\n");
    printf("========5.追加选手数据=========\n");
    printf("========6.删除选手信息=========\n");
    printf("========7.修改选手信息=========\n");
    printf("========8.输出全部选手信息======\n");
    printf("========9.写入数据文件=========\n");
    printf("========0.退出系统============\n");
    printf("=============================\n");
}


  • 运行结果如下:
    在这里插入图片描述

3.输入选手数据模块

//1.输入选手数据
S \*create() {
    S \*head, \*p, \*q;//定义指针
    int i;
    head = (S \*) malloc(sizeof(S));//头节点开辟空间
    head->average = 0;//置空头节点的average成员
    head->next = NULL;//置空头节点的指针域
    q = head;//q指针记录头节点的地址
    p = head->next;//p指针记录头节点的指针域的地址
    printf("请输入选手编号和姓名(编号为0则表示停止):\n");
    int id;
    printf("请输入选手id(输入0退出):\n");
    scanf("%d", &id);
    while (id != 0)//输入选手编号输入为零停止循环
    {
        p = (S \*) malloc(sizeof(S));//p指针开辟空间
        p->id = id;
        printf("请输入选手姓名:\n");
        scanf("%s", p->name);
        for (i = 0; i < NUMBER; i++) {
            p->grade[i] = 0;
        }
        p->sum = 0;
        p->average = 0;
        p->next = NULL;//置空p节点的指针域
        q->next = p;//p,q节点连接
        q = p;//q指针后移
        printf("请输入选手id(输入0退出):\n");
        scanf("%d", &id);
    }
    return head;//返回链表的起始地址
}


  • 运行结果如下:
    在这里插入图片描述

4.评委打分模块

//2.评委打分
void giveScore(S \*head) {
    //定义指针
    S \*p;
    p = head->next;
    float max, min;
    while (p != NULL) {
        if (p->grade[0] != 0) {
            p = p->next;
            continue;
        }
        printf("请十位评为给编号为%d的选手打分:\n", p->id);
        p->sum = 0;
        for (int i = 0; i < NUMBER; i++) {
            printf("请第%d位评委打分:", i);
            scanf("%f", &p->grade[i]);
            p->sum += p->grade[i];
        }
        printf("\n");
        min = max = p->grade[0];
        for (int i = 0; i < NUMBER; i++) {
            if (p->grade[i] > max)
                max = p->grade[i];
            if (p->grade[i] < min)
                min = p->grade[i];
        }
        p->average = (p->sum - max - min) / 8;
        p = p->next;
    }
}


  • 运行结果如下:
    在这里插入图片描述

5.按照平均成绩排序 – 采用冒泡排序,交换节点

//3.按照平均成绩排序 -- 采用冒泡排序,交换节点
void sort(S \*head) {
    S \*p, \*pre, \*temp, \*tail;

    tail = NULL;
// 算法的核心部分,节点交换
    while (head->next != tail) {
        pre = head;
        p = head->next;
        while (p->next != tail) {
            if (p->average < p->next->average) {
                temp = p->next;
                pre->next = p->next;
                p->next = p->next->next;
                pre->next->next = p;
                p = temp;
            }
            // 节点后移
            p = p->next;
            pre = pre->next;
        }
        tail = p;
    }
}


6.数据查询模块

//4.根据id查找节点模块
void search(S \*head) {
    S \*p;//定义指针
    int id;//定义b用于输入查找编号
    printf("请输入要查找的选手编号:");
    //输入查找编号
    scanf("%d", &id);
    p = head->next;
    while (p != NULL) {
        if (p->id == id)//判断是否找到选手编号
        {
            //为真时,输出信息
            printf("编号\t姓名\t\t\t\t\t十位评委的成绩\t\t\t\t\t总成绩\t平均成绩\n");
            printf("%d %s %.2f%.2f%.2f%.2f%.2f%.2f%.2f%.2f%.2f%.2f %.2f %.2f\n", p->id, p->name, p->grade[0],
                   p->grade[1], p->grade[2], p->grade[3], p->grade[4], p->grade[5], p->grade[6], p->grade[7],
                   p->grade[8], p->grade[9], p->sum, p->average);
            break;
        } else
            //为假时
            p = p->next;//指针后移
    }
    if (p == NULL)//查找到最后一个节点还未查到要的编号时,输出ERROR INPUT
        printf("输入的选手编号错误\n");
}

  • 运行结果如下:
    在这里插入图片描述

7.追加选手数据模块

//5.追加节点模块(可多个插入)
void insert(S \*head) {
    int i, id, flag = 1;//flag实现判断指针是否到达最后一个节点
    S \*p, \*q, \*r;    //定义指针便于插入操作
    p = head;
    printf("请输入选手信息:\n");
    printf("请输入选手id(输入0退出):\n");
    scanf("%d", &id);
    while (id != 0)//输入编号不为零时循环,以零终止,可实现多个插入
    {
        r = (S \*) malloc(sizeof(S));//为r开辟空间
        r->next = NULL;//置空r的指针域
        //输入相关数据,并计算相关数据
        r->id = id;
        printf("请输入选手姓名:\n");
        scanf("%s", r->name);
        r->sum = 0;
        r->average = 0;
        for (i = 0; i < NUMBER; i++) {
            r->grade[i] = 0;
        }

        while (p->next != NULL) {
            p = p->next;
        }
        p->next = r;

        printf("请输入选手id(输入0退出):\n");
        scanf("%d", &id);
    }
}


  • 运行结果如下:
    在这里插入图片描述

8.删除选手信息模块

//6.删除节点模块
void del(S \*head) {
    S \*p, \*q;//定义指针
    int b;//用于输入编号查找删除
    p = head;//p记录头节点的地址
    q = head->next;//q记录头节点的指针域的地址
    printf("请输入要删除选手的id:");
    //输入编号
    scanf("%d", &b);
    while (q != NULL)//q不为空时执行循环
    {
        if (q->id == b)//判断是否找到输入的编号
            //为真时
        {
            p->next = q->next;//断开q节点
            free(q);//释放q节点neicun
            q = NULL;    //置空q指针防止出现野指针
        } else {
            //判断为假时
            p = p->next;//p指针后移
            q = q->next;//q指针后移
        }
    }
    if (p == NULL)//当查找到最后一个节点还未查到要删除的编号时,输出ERROR INPUT
        printf("输入的选手id不存在!\n");
}

  • 运行结果如下:
    在这里插入图片描述

9.修改选手信息模块

//修改选手信息
void update(S \*head) {
    S \*p, \*q, \*new;//定义指针
    int id;//用于输入编号查找删除
    float min = 999, max = 0;
    p = head;//p记录头节点的地址
    q = head->next;//q记录头节点的指针域的地址
    printf("请输入要修改的选手id:");
    //输入编号
    scanf("%d", &id);
    while (q != NULL)//q不为空时执行循环
    {
        //判断是否找到输入的编号
        if (q->id == id) {
            new = (S \*) malloc(sizeof(S));
            printf("请输入%d号选手的新成绩:", q->id);
            new->id = q->id;
            strcpy(new->name, q->name);
            new->sum = 0;
            new->average = 0;
            for (int i = 0; i < NUMBER; ++i) {
                scanf("%f", &new->grade[i]);
                new->sum += new->grade[i];
                if (new->grade[i] > max) {
                    max = new->grade[i];
                }
                if (new->grade[i] < min) {
                    min = new->grade[i];
                }
            }
            new->average = (new->sum - max - min) / (NUMBER - 2);
            new->next = q->next;//断开q节点
            p->next = new;
            free(q);//释放q节点
            q = NULL;    //置空q指针防止出现野指针
        } else {
            //判断为假时
            p = p->next;//p指针后移
            q = q->next;//q指针后移
        }
    }
    if (p == NULL)//当查找到最后一个节点还未查到要修改的编号时
        printf("输入的选手id不存在!\n");
}


  • 运行结果如下:
    在这里插入图片描述

10.输出全部选手信息模块

//7.输出全部选手信息
void print(S \*head) {
    int i;
    S \*p = head->next;
    printf("编号\t姓名\t总成绩\t平均成绩\t\t\t\t\t十位评委的成绩\t\t\t\t\n");
    while (p)//当p不为空的时候执行
    {
        printf("%-d\t%-s\t%-.2f\t%-.2f\t", p->id, p->name, p->sum, p->average);
        for (i = 0; i < NUMBER; i++)
            printf("%-6.2f\t", p->grade[i]);
        printf("\n");
        p = p->next;//指针后移
    }
}


  • 运行结果如下:
    在这里插入图片描述

11.写入数据文件(记得修改写入文件地址)

//8.写入数据文件
void save(S \*head) {
    S \*p;
    p = head->next;
    FILE \*fp;
    if ((fp = fopen("C:\\Users\\zhuxuanyu\\Desktop\\file.txt", "wt")) == NULL) {
        printf("打开写入文件fle.txt失败\n");
        return;
    }
    while (p) {
        fprintf(fp, "编号:%-d\t 姓名:%-s\t 总成绩:%-.2f\t 平均成绩:%-.2f\t 评委打分:", p->id, p->name, p->sum,
                p->average);
        for (int i = 0; i < NUMBER; ++i) {
            fprintf(fp, "%-.2f\t", p->grade[i]);
        }
        fprintf(fp, "\n");
        p = p->next;
    }
    fclose(fp);
}

五、完整代码


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
//创建结构体及其成员 

#define MAXSIZE 20
#define NUMBER 10

typedef struct Node {
    int id;//编号
    char name[MAXSIZE];//姓名
    float grade[NUMBER];//评委评分
    float sum;//总分
    float average;//平均分
    struct Node \*next;//指针域
} S;//结构体定义为S
//各函数定义
void menu(); //菜单函数 
S \*create();//创建链表函数 
void print(S \*);//输出链表函数 
void insert(S \*);//插入节点函数 
void del(S \*);//删除节点函数
void update(S \*);

void search(S \*);//查找节点函数
void sort(S \*);//节点排序函数 
void save(S \*);

void giveScore(S \*);


//主函数
int main() {
    S \*head;
    int n, a = 1;//n用来控制选择操作类型,a控制循环,以-1终止


![img](https://img-blog.csdnimg.cn/img_convert/ee864a13836278a5a3af92094dd81490.png)
![img](https://img-blog.csdnimg.cn/img_convert/dde37275d9aa96dd1074cff79b42f432.png)

**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**


**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

oid menu(); //菜单函数 
S \*create();//创建链表函数 
void print(S \*);//输出链表函数 
void insert(S \*);//插入节点函数 
void del(S \*);//删除节点函数
void update(S \*);

void search(S \*);//查找节点函数
void sort(S \*);//节点排序函数 
void save(S \*);

void giveScore(S \*);


//主函数
int main() {
    S \*head;
    int n, a = 1;//n用来控制选择操作类型,a控制循环,以-1终止


[外链图片转存中...(img-CJfvvSnZ-1714457783062)]
[外链图片转存中...(img-eAeh9wQm-1714457783063)]

**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**


**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值