【C++课程设计——演讲比赛系统】

文章目录

  • 前言
  • 一、演讲比赛程序需求
  • 二、每个功能模块的实现
    • 1. 创建管理类(.h文件)
    • 2.1. 创建管理类(.cpp文件)
    • 3.创建参赛选手类(.h)
    • 4.将整体逻辑进行封装
  • 测试
  • 项目总结


前言

在学习完C++的stl容器后,我们来写一下小项目对其进行应用!
项目名称为:演讲比赛系统

一、演讲比赛程序需求

  1. 比赛规则
    学校举行一场演讲比赛,共有12个人参加。比赛共两轮,第一轮为淘汰赛,第二轮为决赛。
    比赛方式:分组比赛,每组6个人;选手每次要随机分组,进行比赛
    每名选手都有对应的编号,如 10001 ~ 10012
    第一轮分为两个小组,每组6个人。 整体按照选手编号进行抽签后顺序演讲。
    当小组演讲完后,淘汰组内排名最后的三个选手,前三名晋级,进入下一轮的比赛。
    第二轮为决赛,前三名胜出
    每轮比赛过后需要显示晋级选手的信息

  2. 程序功能
    开始演讲比赛:完成整届比赛的流程,每个比赛阶段需要给用户一个提示,用户按任意键后继续下一个阶段
    查看往届记录:查看之前比赛前三名结果,每次比赛都会记录到文件中,文件用.csv后缀名保存
    清空比赛记录:将文件中数据清空
    退出比赛程序:可以退出当前程序

二、每个功能模块的实现

1. 创建管理类(.h文件)

功能描述:
提供菜单界面与用户交互
对演讲比赛流程进行控制
与文件的读写交互

#include<iostream>
#include<vector>
#include<deque>
#include<string>
#include<map>
#include"Speaker.h"
#include<time.h>
#include<algorithm>
#include<numeric>
#include<fstream>
using namespace std;

class speechManager {
public:

	//构造函数
	speechManager();

	//菜单函数
	void Show_Menu();

	//初始化函数
	void initSperker();

	//创建12位初始选手
	void creatSpeaker();


	//开始进行比赛
	void startSpeech();

	//比赛抽签
	void speechDraw();

	//开始比赛
	void speechContest();

	//显示晋级结果
	void showScore();
	
	//保存每届记录
	void saveRecord();
	
	//退出系统
	void gameOver();
	
	//析构函数
	~speechManager();

	//读取记录
	void loadRecord();
	//展示记录
	void showRecord();

	//清空文件
	void clearRecord();
	//文件为空的标志
	bool fileIsEmpty;
	//往届记录
	map<int, vector<string>> _Record;

	vector<int> v1;//存放开始所有参赛选手

	vector<int> v2;//存放第一轮结束后晋级选手

	vector<int> vVictory;//存放最终前三名

	map<int, Speaker> _Speaker;

	int index;//记录该次比赛轮数
};

2.1. 创建管理类(.cpp文件)

#include"SpeechManager.h"

//构造函数
speechManager::speechManager()
{
	initSperker();
	creatSpeaker();
	this->loadRecord();
}

//菜单函数
void speechManager::Show_Menu()
{
	cout << "************************************************" << endl;
	cout << "*************欢迎参加演讲比赛*******************" << endl;
	cout << "*************1.开始演讲比赛*********************" << endl;
	cout << "*************2.查看往届比赛记录*****************" << endl;
	cout << "*************3.清空比赛记录*********************" << endl;
	cout << "*************0.退出比赛系统*********************" << endl;
	cout << "************************************************" << endl;
}


//退出系统
void speechManager::gameOver()
{
	cout << "欢迎下次再来!" << endl;
	system("pause");
	system("cls");
}



//初始化函数
void speechManager::initSperker()
{
	v1.clear();
	v2.clear();
	vVictory.clear();
	_Speaker.clear();
	index = 1;
	this->_Record.clear();
}


//创建12位初始选手
void speechManager::creatSpeaker()
{
	string nameSeed = "ABCDEFGHIJKL";
	for (int i = 0; i < nameSeed.size(); i++)
	{
		string name("选手");
		name += nameSeed[i];
		Speaker sp;
		sp._name = name;
		for (int i = 0; i < 2; i++)
		{
			sp._score[i] = 0;
		}
		v1.push_back(10001 + i);
		_Speaker.insert(make_pair(10001 + i, sp));
	}
}


//比赛抽签
void speechManager::speechDraw()
{
	cout << "第 " << index << " 轮选手正在抽签" << endl;
	cout << "-----------------------------" << endl;
	cout << "抽签后的演讲顺序如下" << endl;

	if (index == 1)
	{
		random_shuffle(v1.begin(), v1.end());

		for (auto it = v1.begin(); it != v1.end(); it++)
		{
			cout << *it << " ";
		}
		cout << endl;
	}
	else
	{
		random_shuffle(v2.begin(), v2.end());

		for (auto it = v2.begin(); it != v2.end(); it++)
		{
			cout << *it << " ";
		}
		cout << endl;
	}
	cout << "-----------------------------------" << endl;
	system("pause");
}

//开始比赛
void speechManager::speechContest()
{
	cout << "-------------第" << index << "轮比赛正式开始---------------" << endl;
	multimap<double, int, greater<double>> groupScore;//临时容器,可排出分数由高到低的选手编号
	//并且此容器遍历为出来为降序

	int num = 0;
	vector<int> v_src;
	if (index == 1)
	{
		v_src = v1;
	} 
	else
	{
		v_src = v2;
	}

	for (auto it = v_src.begin(); it != v_src.end(); it++)
	{
		num++;

		deque<double> d;
		for (int i = 0; i < 10; i++)//10个评委打分
		{
			double score = (rand() % 401 + 600) / 10.0f;
			d.push_back(score);
		}

		谁有争议,可以查看分数
		//for (auto e : d)
		//{
		//	cout << e << " ";
		//}
		//cout << endl;

		sort(d.begin(), d.end());
		d.pop_back();
		d.pop_front();
		double sum = accumulate(d.begin(), d.end(), 0.0f);

		//double avg = sum / (double)d.size();
		double avg = sum /d.size();

		_Speaker[*it]._score[index - 1] = avg;

		groupScore.insert(make_pair(avg, *it));

		if (num % 6 == 0)
		{
			cout << "第" << num / 6 << "组比赛名次" << endl;

			for (auto it = groupScore.begin(); it != groupScore.end(); it++)
			{
				cout << "编号: " << it->second << " 姓名: " << this->_Speaker[it -> second]._name 
					<< " 成绩: " << this->_Speaker[it->second]._score[this->index - 1] <<endl;
			}

			int count = 0;
			//取前三名
			for (auto it = groupScore.begin(); it!= groupScore.end() && count < 3; it++, count++)
			{
				if (this->index == 1)
				{
					v2.push_back((*it).second);
				}
				else
				{
					vVictory.push_back((*it).second);
				}
			}
			groupScore.clear();
			cout << endl;
		}
	}
	cout << "---------第" << index << "轮比赛完毕-----------" << endl;
	system("pause");
}

//显示晋级结果
void speechManager::showScore()
{
	cout << "第" << index << "轮晋级选手如下" << endl;
	vector<int> v;
	if (index == 1)
	{
		v = v2;
	}
	else
	{
		v = vVictory;
	}
	for (auto it = v.begin(); it != v.end(); it++)
	{
		cout << "选手编号:" << *it << " 姓名: " << _Speaker[*it]._name << " 得分: " <<
			_Speaker[*it]._score[this->index - 1] << endl;
	}
	cout << endl;
	//system("cls");
	//this->Show_Menu();
}


//开始进行比赛
void speechManager::startSpeech()
{
	//第一轮抽签
	speechDraw();
	//第一轮比赛
	speechContest();
	//显示晋级结果
	showScore();
	index++;
	//第二轮抽签
	speechDraw();
	//第二轮比赛
	speechContest();
	//显示最终结果
	showScore();
	//保存结果
	saveRecord();
	//重置属性
	initSperker();
	creatSpeaker();
	this->loadRecord();
	system("cls");
}

//保存每届记录
void speechManager::saveRecord()
{
	ofstream ofs;
	ofs.open("speech.csv", ios::out | ios::app);//csv格式需要添加逗号分隔开

	for (auto it = vVictory.begin(); it != vVictory.end(); it++)
	{
		ofs << *it << "," << _Speaker[*it]._score[1] << ",";
	}
	ofs << endl;

	ofs.close();

	//cout << "记录已保存" << endl;
	this->fileIsEmpty = false;
	system("pause");
}


//读取记录
void speechManager::loadRecord()
{
	ifstream ifs("speech.csv", ios::in); //输入流对象 读取文件

	/*ifstream ifs;
	ifs.open("speech.csv", ios::in);*/

	if (!ifs.is_open())
	{
		this->fileIsEmpty = true;
		///cout << "文件不存在!" << endl;
		ifs.close();
		return;
	}
	char ch;
	ifs >> ch;
	if (ifs.eof())//如果读到了文件结束标志
	{
		//cout << "文件为空!" << endl;
		this->fileIsEmpty = true;
		ifs.close();
		return;
	}
	//文件不为空
	this->fileIsEmpty = false;
	ifs.putback(ch); //读取的单个字符放回去
	string data;
	int index = 0;
	while (ifs >> data)
	{
		vector<string>v;
		int pos = -1;
		int start = 0;
		while (true)
		{
			pos = data.find(",", start); //从0开始查找 ','
			if (pos == -1)
			{
				break; //找不到break返回
			}
			string tmp = data.substr(start, pos - start); //找到了,进行分割 参数1 起始位置,参数2 截取长度
			v.push_back(tmp);
			start = pos + 1;
		}
		this->_Record.insert(make_pair(index, v));
		index++;
	}
	ifs.close();
}


void speechManager::showRecord()
{
	if (this->fileIsEmpty)
	{
		cout << "文件为空或不存在!" << endl;
	}
	else
	{
		for (int i = 0; i < this->_Record.size(); i++)
		{
			cout << "第" << i + 1 << "届 " <<
				"冠军编号:" << this->_Record[i][0] << " 得分:" << this->_Record[i][1] << " "
				"亚军编号:" << this->_Record[i][2] << " 得分:" << this->_Record[i][3] << " "
				"季军编号:" << this->_Record[i][4] << " 得分:" << this->_Record[i][5] << endl;
		}
	}
	
	system("pause");
	system("cls");
}


void speechManager::clearRecord()
{
	cout << "确认清空?" << endl;
	cout << "1、确认" << endl;
	cout << "2、返回" << endl;
	int select = 0;
	cin >> select;
	if (select == 1)
	{
		//打开模式 ios::trunc 如果存在删除文件并重新创建
		ofstream ofs("speech.csv", ios::trunc);
		ofs.close();
		//初始化属性
		this->initSperker();
		//创建选手
		this->creatSpeaker();
		//获取往届记录
		this->loadRecord();
		cout << "清空成功!" << endl;
	}
	system("pause");
	system("cls");
}

//析构函数
speechManager::~speechManager()
{

}

3.创建参赛选手类(.h)

#include<string>
#include<iostream>
using namespace std;
class Speaker {
public:
	string _name;

	double _score[2];

 };

4.将整体逻辑进行封装

#include"SpeechManager.h"

int main0()
{
	speechManager pm;
	srand((unsigned int)(time(NULL)));
	int choice = 1;
	do 
	{
		pm.Show_Menu();
		cout << "请输入你的选项:" << endl;
		cin >> choice;
		switch (choice)
		{
		case 1:
			pm.startSpeech();
			break;
		case 2:
			pm.showRecord();
			break;
		case 3:
			pm.clearRecord();
			break;
		case 0:
			pm.gameOver();
			break;
		default:
			cout << "输入有误,请按任意键后重新输入!" << endl;
			system("pause");
			system("cls");
			break;
		}

	} while (choice);

	return 0;
}

测试

开始进行比赛,进行抽签

在这里插入图片描述

第一轮比赛结束

在这里插入图片描述

第二轮比赛开始抽号

在这里插入图片描述

第二轮比赛结束

在这里插入图片描述

第二轮比赛得出结果

在这里插入图片描述

查看往届比赛成绩

在这里插入图片描述

在这里插入图片描述

清空往届比赛成绩

在这里插入图片描述

在这里插入图片描述

项目总结

本案例主要是对C++的stl容器和文件操作进行使用,基本涵盖了容器常用接口的使用,对大家对于stl的进一步熟悉能起较大左右作用!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/755146.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

常见的反爬手段和解决思路(爬虫与反爬虫)

常见的反爬手段和解决思路&#xff08;爬虫与反爬虫&#xff09; 学习目标1 服务器反爬的原因2 服务器长反什么样的爬虫&#xff08;1&#xff09;十分低级的应届毕业生&#xff08;2&#xff09;十分低级的创业小公司&#xff08;3&#xff09;不小心写错了没人去停止的失控小…

排序算法(1)之插入排序----直接插入排序和希尔排序

个人主页&#xff1a;C忠实粉丝 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 C忠实粉丝 原创 排序之插入排序----直接插入排序和希尔排序(1) 收录于专栏【数据结构初阶】 本专栏旨在分享学习数据结构学习的一点学习笔记&#xff0c;欢迎大家在评论区交流讨…

JavaParser抽取测试用例对应的被测方法

背景介绍 博主目前要做的工作需要将一个java项目的所有RD手写的测试用例和被测方法对应起来&#xff0c;最后将得到的结果存入一个json文件。 本教程以项目GitHub - binance/binance-connector-java 为例。 结果展示 最终会得到一个 funcTestMap.json&#xff0c;里面存放着…

昇思25天学习打卡营第6天|linchenfengxue

​​​​​​SSD目标检测 SSD&#xff0c;全称Single Shot MultiBox Detector&#xff0c;是Wei Liu在ECCV 2016上提出的一种目标检测算法。使用Nvidia Titan X在VOC 2007测试集上&#xff0c;SSD对于输入尺寸300x300的网络&#xff0c;达到74.3%mAP(mean Average Precision)以…

kafka-Stream详解篇(附案例)

文章目录 Kafka Stream 概述Kafka Stream 概念Kafka Stream 数据结构入门案例一需求描述与分析配置KafkaStream定义处理流程声明Topic接收处理结果发送消息测试 入门案例二需求描述与分析定义处理流程接收处理结果声明Topic 更多相关内容可查看 Kafka Stream 概述 Kafka Strea…

脉冲同步器(快到慢)

目录 描述 输入描述&#xff1a; 输出描述&#xff1a; 参考代码 描述 sig_a 是 clka&#xff08;300M&#xff09;时钟域的一个单时钟脉冲信号&#xff08;高电平持续一个时钟clka周期&#xff09;&#xff0c;请设计脉冲同步电路&#xff0c;将sig_a信号同步到时钟域 cl…

Excel 宏录制与VBA编程 —— 15、MsgBox参数详解

Msgbox参数具体如下 Msgbox参数使用1 Msgbox参数使用2&#xff08;返回值示例&#xff09; &ensp ;###### 关注 笔者 - jxd

Vue 项目运行时,报错Error: Cannot find module ‘node:path‘

Vue 项目运行时&#xff0c;报错Error: Cannot find module ‘node:path’ internal/modules/cjs/loader.js:883throw err;^Error: Cannot find module node:path Require stack: - D:\nodejs\node_modules\npm\node_modules\node_modules\npm\lib\cli.js - D:\nodejs\node_mo…

GMSB文章七:微生物整合分析

欢迎大家关注全网生信学习者系列&#xff1a; WX公zhong号&#xff1a;生信学习者Xiao hong书&#xff1a;生信学习者知hu&#xff1a;生信学习者CDSN&#xff1a;生信学习者2 介绍 本文通过多元方差分析和典型相关分析研究微生物&#xff08;species&#xff09;、细胞因子…

【面试干货】与的区别:位运算符与逻辑运算符的深入探讨

【面试干货】&与&&的区别&#xff1a;位运算符与逻辑运算符的深入探讨 1、&&#xff1a;位运算符2、&&&#xff1a;逻辑运算符3、&与&&的区别 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; & 和 …

NIVision-LabVIEW在灰度图上画圆

问题来源 在csdn上看到的这样一个问题&#xff0c;好像也没个正经答案&#xff0c;都用chatGPT回答&#xff0c;挺没劲的。不说提供个vi源代码&#xff0c;至少也来张截图嘛。我想着问题也不难&#xff0c;就自己动动手吧。 代码展示1 1、首先使用imaq ArrayToImage.vi创建了一…

《昇思25天学习打卡营第01天|sun65535》

开始 昇思25天打卡训练营&#xff0c;让我第一次了解了华为昇思的平台&#xff0c;之前也有自己本地使用4060训练了一些“小模型”&#xff0c;但是都是比较皮毛的知识&#xff0c;只是根据教程去搭建。很少了解到具体的过程。昇思25天打卡训练营给了一个比较全面的训练课程。…

IoTDB Committer+Ratis PMC Member:“两全其美”的秘诀是?

IoTDB & Ratis 双向深耕&#xff01; 还记得一年前我们采访过拥有 IoTDB 核心研发 Ratis Committer “双重身份”的社区成员宋子阳吗&#xff1f;&#xff08;点此阅读&#xff09; 我们高兴地发现&#xff0c;一年后&#xff0c;他在两个项目都更进一步&#xff0c;已成为…

Firefox 编译指南2024 Windows10- 定制化您的Firefox(四)

1. 引言 定制化您的Firefox浏览器是一个充满乐趣且富有成就感的过程。在2024年&#xff0c;Mozilla进一步增强了Firefox的灵活性和可定制性&#xff0c;使得开发者和高级用户能够更深入地改造和优化浏览器以满足个人需求。从界面的微调到功能的增强&#xff0c;甚至是核心代码…

vscode 生成项目目录结构 directory-tree 实用教程

1. 安装插件 directory-tree 有中文介绍&#xff0c;极其友好&#xff01; 2. 用 vscode 打开目标项目 3. 快捷键 Ctrl Shift p&#xff0c;输入 Directory Tree 后回车 会在 README.md 文件的底部生成项目目录&#xff08;若项目中没有 README.md 文件&#xff0c;则会自动创…

casefold()方法——所有大写字符转换为小写

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 语法参考 casefold()方法是Python3.3版本之后引入的&#xff0c;其效果和lower()方法非常相似&#xff0c;都可以转换字符串中所有大写字符为小写。…

windows MSVC编译安装libcurl

$ git clone https://github.com/curl/curl.git $ cd curl/winbuild依照curl/winbuild/README.md的指示&#xff0c; 启动visual studio的命令行工具&#xff0c;这里要注意别选错. 如果要编译出x64版本的libcurl&#xff0c;就用x64的命令行工具&#xff1b;如果要编译出x86…

怎么修改钻孔表的大小?

导入 在Cadence中最后要生成Gerber文件交由板厂制版时&#xff0c;其中有个提取钻孔表的过程。以往的过程并没有对钻孔表要求&#xff0c;今天却要修改钻孔表的大小了&#xff0c;如何做呢&#xff1f;这是一个非常罕见的操作&#xff0c;特此记录。 原理 1、先来复习一下如何…

使用AI的100种方法#翻译神器N3

Text "100 ways to" and "use AI" in the poster center .A cozy desk setup with an open notebook featuring notes and drawings, a cup of coffee, a white pen, and dried flowers. Warm, earthy tones create a calming, aesthetic vibe. 第 3 种可能…

量产工具一一显示系统(一)

目录 前言 一、项目介绍和应用 1.简单易用 2.软件可配置、易扩展 3.纯 C 语言编程 4.类似界面应用 二、项目总体框架 三、显示系统 1.显示系统数据结构抽象 &#xff08;1&#xff09;common.h &#xff08;2&#xff09;disp_manager.h 2.Framebuffer编程 &#x…