【演讲比赛流程管理系统(C++版)】

news2024/11/25 22:56:27

一、演讲比赛程序需求

1.1、比赛规则

·学校举行一场演讲比赛,共有12个人参加。比赛共两轮,第一轮为淘汰赛,第二轮为决赛

·每名选手都有对应的编号,如10001~10012

·比赛方式:分组比赛,每组6个人

·第一轮分为两个小组,整体按照选手编号进行抽签后顺序演讲。

·十个评委分别给每名选手打分,去除最高分和最低分,求的平均分为本轮选手的成绩 

·当小组演讲完后,淘汰组内排名最后的三个选手,前三名晋级,进入下一轮的比赛。 

·第二轮为决赛,前三名胜出 

·每轮比赛过后需要显示晋级选手的信息 

1.2、程序功能 

·开始演讲比赛:完成整届比赛的流程,每个比赛阶段需要给用户一个提示,用户按任意键后继续下一个阶段

·查看往届记录:查看之前比赛前三名结果,每次比赛都会记录到文件中,文件用.csv后缀名保存

·清空比赛记录:将文件中数据清空

·退出比赛程序:可以退出当前程序

1.3、程序效果图

二、创建管理类

功能描述:

·提供菜单界面与用户交互

·对演讲比赛流程进行控制

·与文件的读写交互

2.1、创建文件 

·在头文件和源文件的文件夹下分别创建speechManager.h和speechManager.cpp文件 

2.2、头文件实现 

在speechManager.h中设计管理类

代码如下: 

#pragma once
using namespace std;
#include <iostream>

//设计演讲管理类
class SpeechManager
{
public:
	//构造函数
	SpeechManager();

	//析构函数
	~SpeechManager();
};

2.3、源文件实现

在speechManager.cpp中将构造和析构函数空实现补全

#include "SpeechManager.h"

//构造函数
SpeechManager::SpeechManager()
{
	
}

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

}

三、菜单功能

功能描述:与用户的沟通界面

3.1、添加成员函数 

在管理类speechManager.h中添加成员函数void show_Menu();

3.2、菜单功能实现 

·在管理类speechManager.cpp中实现show_Menu()函数

//菜单功能
void SpeechManager::show_Menu()
{
	cout << "************************************" << endl;
	cout << "********  欢迎参加演讲比赛  ********" << endl;
	cout << "********   1.开始演讲比赛   ********" << endl;
	cout << "********   2.查看往届记录   ********" << endl;
	cout << "********   3.清空比赛记录   ********" << endl;
	cout << "********   0.退出比赛程序   ********" << endl;
	cout << "************************************" << endl;
	cout << endl;
}

3.3、测试菜单功能

·在演讲比赛流程管理系统.cpp中测试菜单功能

代码: 

using namespace std;
#include <iostream>
#include"speechManager.h"

int main()
{
	//创建管理类对象
	SpeechManager sm;

	sm.show_Menu();

	system("pause");
	return 0;
}

·运行效果如图:

·菜单界面搭建完毕 

四、退出功能 

功能描述:实现退出程序

4.1、提供功能接口

·在main函数中提供分支选择,提供每个功能接口 

代码:

using namespace std;
#include <iostream>
#include"speechManager.h"

int main()
{
	//添加随机数种子
	srand((unsigned int)time(NULL));

	//创建管理类对象
	SpeechManager sm;

	int choice = 0;//用于存储用户输入
	while (true)
	{
		sm.show_Menu();

		cout << "请输入您的选择:";
		cin >> choice;

		switch (choice)
		{
		case 1://开始比赛
			sm.startSpeech();
			break;
		case 2://查看往届比赛记录
			sm.showRecord();
			break;
		case 3://清空比赛记录
			sm.clearRecord();
			break;
		case 0://退出系统
			sm.exitSystem();
			break;
		default:
			system("cls");//清屏
			break;
		}
	}

	system("pause");
	return 0;
}

4.2、实现退出功能

在speechManager.h中提供退出系统的成员函数 void exitsystem();

在speechManager.cpp中提供具体的功能实现 

//退出系统
void SpeechManager::exitSystem()
{
	cout << "欢迎下次使用!" << endl;
	system("pause");
	exit(0);
}

4.3、测试功能

在main函数分支0选项中,调用退出程序的接口

运行测试效果如图: 

五、演讲比赛功能 

5.1、功能分析

比赛流程分析:

抽签 → 开始演讲比赛 → 显示第一轮比賽结果 → 抽签 → 开始演讲比賽 → 显示前三名结果 → 保存分数

5.2、创建选手类 

·选手类中的属性包含:选手姓名、分数

·头文件中创建speaker.h文件,并添加代码:

#pragma once
#include <iostream>
using namespace std;

//选手类
class Speaker
{
public:
	string m_Name;//姓名
	double m_Score[2];//分数 - 最多有两轮得分
};

5.3、比赛

5.3.1、成员属性添加

·在speechManager.h中添加属性

#pragma once
using namespace std;
#include <iostream>

//设计演讲管理类
class SpeechManager
{
public:
	//成员属性
	//保存第一轮比赛选手编号的容器
	vector<int> v1;

	//保存第一轮晋级选手编号的容器
	vector<int> v2;

	//保存胜出的前三名选手编号的容器
	vector<int> vVictory;

	//存放编号以及对应具体选手的容器
	map<int, Speaker>m_Speaker;

	//存放比赛轮数
	int m_Index;
};
5.3.2、初始化属性

·在speechManager.h中提供开始比赛的的成员函数void initspeech();

·在speechManager.cpp中实现void initspeech(); 

//初始化容器和属性
void SpeechManager::initSpeech()
{
	//容器都置空
	this->v1.clear();
	this->v2.clear();
	this->vVictory.clear();
	this->m_Speaker.clear();

	//初始化比赛的轮数
	this->m_Index = 1;

	//将记录的容器也清空
	this->m_Record.clear();
}

·SpeechManager构造函数中调用void initspeech();

5.3.3、创建选手 

·在speechManager.h中提供开始比赛的的成员函数void createspeaker(); 

·在speechManager.cpp中实现void createspeaker(); 

//创建12名选手
void SpeechManager::createSpeaker()
{
	string nameSeed = "ABCDEFGHIJKL";
	for (int i = 0; i < nameSeed.size(); i++)
	{
		string name = "选手";
		name += nameSeed[i];

		Speaker sp;
		sp.m_Name = name;
		for (int j = 0; j < 2; j++)
		{
			sp.m_Score[j] = 0;
		}

		//创建选手编号并放入v1容器中
		this->v1.push_back(10001 + i);

		//选手编号以及对应选手放入map容器
		this->m_Speaker.insert(make_pair(i+10001 , sp));
	}
}

·SpeechManager类的构造函数中调用void createspeaker();

·测试在main函数中,可以在创建完管理对象后,使用下列代码测试12名选手初始状态 

for (map<int, Speaker>::iterator it = sm.m_Speaker.begin(); it != sm.m_Speaker.end(); it++)
{
	cout << "选手编号:"<<it->first
		<< " 姓名:" << it->second.m_Name
		<< " 成绩:" << it->second.m_Score[0] << endl;
}

·测试效果如图: 

·测试完毕后,可以将测试代码删除或注释。 

5.3.4、开始比赛成员函数添加 

·在speechManager.h中提供开始比赛的的成员函数void startSpeech(); 

·该函数功能是主要控制比赛的流程 

·在speechManager.cpp中将startSpeech的空实现先写入 

·我们可以先将整个比赛的流程写到函数中 

//开始比赛 - 比赛整个流程控制函数
void SpeechManager::startSpeech()
{
	//第一轮开始比赛
	//1.抽签

	//2.比赛

	//3.显示晋级结果

	//第二轮开始比赛
	//1.抽签

	//2.比赛

	//3.显示最终结果

	//4.保存分数到文件中
}
5.3.5、抽签

功能描述:

·正式比赛前,所有选手的比赛顺序需要打乱,我们只需要将存放选手编号的容器打乱次序即可

·在speechManager.h中提供抽签的的成员函数void speechDraw(); 

·在speechManager.cpp中实现成员函数void speechDraw(); 

//抽签
void SpeechManager::speechDraw()
{
	cout << "第 << " << this->m_Index << " >> 轮比赛选手正在抽签" << endl;
	cout << "-----------------------------" << endl;
	cout << "抽签后演讲顺序如下:" << endl;
	if (this->m_Index == 1)
	{
		//第一轮比赛
		random_shuffle(v1.begin(), v1.end());
		for (vector<int>::iterator it = v1.begin(); it != v1.end(); it++)
		{
			cout << *it << " ";
		}
		cout << endl;
	}
	else
	{
		//第二轮比赛
		random_shuffle(v2.begin(), v2.end());
		for (vector<int>::iterator it = v2.begin(); it != v2.end(); it++)
		{
			cout << *it <<" ";
		}
		cout << endl;
	}
	cout << "-----------------------------" << endl;
	system("pause");
	cout << endl;
}

·在startSpeech比赛流程控制的函数中,调用抽签函数

·在main函数中,分支1选项中,调用开始比赛的接口 

·测试 

5.3.6、开始比赛 

·在speechManager.h中提供比赛的的成员函数void speechContest(); 

·在speechManager.cpp中实现成员函数void speechContest(); 

//比赛
void SpeechManager::speechContest()
{
	cout << "----------第 << " << this->m_Index << " >> 轮比赛正式开始!----------" << endl;

	//准备临时容器存放小组成绩
	multimap<double, int, greater<double>> groupScore;
	int num = 0;//用于记录人员个数 - 6人一组

	vector<int>v_Src;//比赛选手容器
	if (this->m_Index == 1)
	{
		v_Src = v1;
	}
	else
	{
		v_Src = v2;
	}

	//遍历所有选手进行比赛
	for (vector<int>::iterator it = v_Src.begin(); it != v_Src.end(); it++)
	{
		num++;

		//评委打分
		deque<double>d;
		for (int i = 0; i < 10; i++)
		{
			double score = (rand() % 401 + 600)/10.0f;//(600 ~ 1000) - (60 ~ 100)
			//cout << score << " ";
			d.push_back(score);
		}
		//cout << endl;

		sort(d.begin(), d.end(), greater<double>());//排序
		d.pop_front();//去除最高分
		d.pop_back();//去除最低分
		double sum = accumulate(d.begin(), d.end(), 0.0f);//总分
		double avg = sum / (double)d.size();//平均分

		//将平均分放入map容器中
		this->m_Speaker[*it].m_Score[this->m_Index - 1] = avg;

		//将打分的数据放入到临时小组容器中
		groupScore.insert(make_pair(avg, *it));//key是得分,value是具体选手的编号
		//每6人取出前三名
		if (num % 6 == 0)
		{
			cout << "第" << num / 6 << "小组比赛名次:"<< endl;
			for (multimap<double, int, greater<double>>::iterator it = groupScore.begin(); it != groupScore.end(); it++)
			{
				cout << "编号:" << it->second << " 姓名:" << this->m_Speaker[it->second].m_Name <<
					" 成绩:" << this->m_Speaker[it->second].m_Score[this->m_Index - 1] << endl;
			}

			//取走前三名
			int count = 0;
			for (multimap<double, int, greater<double>>::iterator it = groupScore.begin(); it != groupScore.end() && count < 3; it++ , count++)
			{
				if (this->m_Index == 1)
				{
					v2.push_back((*it).second);
				}
				else
				{
					vVictory.push_back((*it).second);
				}
			}
			groupScore.clear();//小组容器清空
			cout << endl;
		}
	}
	cout << "----------第 << " << this->m_Index << " >> 轮比赛结束!----------" << endl;
	system("pause");
}

·在startSpeech比赛流程控制的函数中,调用比赛函数

·再次运行代码,测试比赛 

5.3.7、显示比赛分数

·在speechManager.h中提供显示分数的成员函数void showScore(); 

·在speechManager.cpp中实现成员函数void showScore(); 

//显示得分
void SpeechManager::showScore()
{
	cout << "----------第 << " << this->m_Index << " >> 轮晋级选手信息如下:----------" << endl;

	vector<int>v;
	if (this->m_Index == 1)
	{
		v = v2;
	}
	else
	{
		v = vVictory;
	}

	for (vector<int>::iterator it = v.begin(); it != v.end(); it++)
	{
		cout << "选手编号:" << *it << " 姓名:" << this->m_Speaker[*it].m_Name << " 成绩:"
			<< this->m_Speaker[*it].m_Score[this->m_Index - 1] << endl;
	}

	cout << endl;
	system("pause");
	system("cls");
	this->show_Menu();
}

·在startSpeech比赛流程控制的函数中,调用显示比赛分数函数

·运行代码,测试效果 

5.3.8、第二轮比赛

第二轮比赛流程同第一轮,只是比赛的轮是+1,其余流程不变 

·在startSpeech比赛流程控制的函数中,加入第二轮的流程 

·测试,将整个比赛流程都跑通 

5.4、保存分数 

功能描述:
·将每次演讲比赛的得分记录到文件中 

功能实现:
·在speechManager.h中添加保存记录的成员函数void saveRecord(); 

·在speechManager.cpp中实现成员函数void saveRecord(); 

//保存分数记录
void SpeechManager::saveRecord()
{
	ofstream ofs;
	ofs.open("speech.csv", ios::out | ios::app);//用追加的方式写文件

	//将每个选手的数据写入文件中
	for (vector<int>::iterator it = vVictory.begin(); it != vVictory.end(); it++)
	{
		ofs << *it << "," << this->m_Speaker[*it].m_Score[1] << ",";
	}
	ofs << endl;

	//关闭文件
	ofs.close();
	cout << "记录已经保存!" << endl;

	//更改文件不为空状态
	this->fileIsEmpty = false;
}

·在startSpeech比赛流程控制的函数中,最后调用保存记录分数函数

·测试,整个比赛完毕后记录保存情况 

·利用记事本打开文件speech.csv,里面保存了前三名选手的编号以及得分 

·至此,整个演讲比赛功能制作完毕! 

六、查看记录

6.1、读取记录分数

·在speechManager.h中添加保存记录的成员函数void loadRecord();

·添加判断文件是否为空的标志bool fileIsEmpty; 

·添加往届记录的容器map<int,vector<string>>m_Record; 

其中m_Record 中的key代表第几届,value记录具体的信息 

·在speechManager.cpp中实现成员函数void loadRecord(); 

//读取记录
void SpeechManager::loadRecord()
{
	ifstream ifs("speech.csv", ios::in);//读文件
	if (!ifs.is_open())
	{
		this->fileIsEmpty = true;
		ifs.close();
		return;
	}

	//文件清空情况
	char ch;
	ifs >> ch;
	if (ifs.eof())
	{
		this->fileIsEmpty = true;
		ifs.close();
		return;
	}

	//文件不为空
	this->fileIsEmpty = false;
	ifs.putback(ch);//将上面读取的单个字符在放回文件

	int index = 0;
	string data;

	while (ifs >> data)
	{
		vector<string>v;//存放6个string字符串

		int pos = -1;//查到","位置的变量
		int start = 0;

		while (true)
		{
			pos = data.find(",", start);
			if (pos == -1)
			{
				//未找到情况
				break;
			}
			string temp = data.substr(start, pos - start);
			v.push_back(temp);
			start = pos + 1;
		}
		this->m_Record.insert(make_pair(index, v));
		index++;
	}
	ifs.close();
}

·在SpeechManager构造函数中调用获取往届记录函数

6.2、查看记录功能 

·在speechManager.h中添加保存记录的成员函数void showRecord();

·在speechManager.cpp中实现成员函数void showRecord(); 

//显示往届记录
void SpeechManager::showRecord()
{
	for (int i = 0; i < this->m_Record.size(); i++)
	{
		cout << "第" << i + 1 << "届 " << "冠军编号:" << this->m_Record[i][0]
			<< " 得分:" << this->m_Record[i][1] <<
			" 亚军编号:" << this->m_Record[i][2]
			<< " 得分:" << this->m_Record[i][3] <<
			" 季军编号:" << this->m_Record[i][4]
			<< " 得分:" << this->m_Record[i][5] << endl;
	}

	system("pause");
	system("cls");
}

6.3、 bug解决

目前程序中有几处bug未解决: 

1.查看往届记录,若文件不存在或为空,并未提示
解决方式:在showRecord函数中,开始判断文件状态并加以判断 

2.若记录为空或不存在,比完赛后依然提示记录为空

解决方式:saveRecord中更新文件为空的标志 

3.比完赛后查不到本届比赛的记录,没有实时更新

解决方式:比赛完毕后,所有数据重置 

4.在初始化时,没有初始化记录容器

解决方式:initSpeech中添加初始化记录容器 

5.每次记录都是一样的
解决方式:在main函数一开始添加随机数种子 

所有bug解决后测试: 

七、清空记录 

7.1、清空记录功能实现

·在speechManager.h中添加保存记录的成员函数void clearRecord();

·在speechManager.cpp中实现成员函数void clearRecord(); 

//清空文件
void SpeechManager::clearRecord()
{
	cout << "是否确认清空文件(Y/N):";
	char select = 0;
	cin >> select;

	if (select == 'Y' || select == 'y')
	{
		//确认清空
		ofstream ofs("speech.csv", ios::trunc);
		ofs.close();

		//初始化容器和属性
		this->initSpeech();

		//创建12名选手
		this->createSpeaker();

		//加载往届记录
		this->loadRecord();

		cout << "清空成功!" << endl;
	}
	system("pause");
	system("cls");
}

7.2、测试清空

在main函数分支3选项中,调用清空比赛记录的接口 

运行程序,测试清空记录:

speech.csv中记录也为空

·至此本案例结束! 

八、源代码

8.1、speaker.h 

#pragma once
#include <iostream>
using namespace std;

//选手类
class Speaker
{
public:
	string m_Name;//姓名
	double m_Score[2];//分数 - 最多有两轮得分
};

 8.2、speechManager.h

#pragma once
using namespace std;
#include <iostream>
#include <vector>
#include <map>
#include "speaker.h"
#include <algorithm>
#include <deque>
#include<functional>
#include <numeric>
#include<fstream>

//设计演讲管理类
class SpeechManager
{
public:
	//构造函数
	SpeechManager();

	//菜单功能
	void show_Menu();

	//退出系统
	void exitSystem();

	//析构函数
	~SpeechManager();

	//初始化容器和属性
	void initSpeech();

	//创建12名选手
	void createSpeaker();

	//开始比赛 - 比赛整个流程控制函数
	void startSpeech();

	//抽签
	void speechDraw();

	//比赛
	void speechContest();

	//显示得分
	void showScore();

	//保存分数记录
	void saveRecord();

	//读取记录
	void loadRecord();

	//显示往届记录
	void showRecord();

	//清空文件
	void clearRecord();

	//判断文件是否为空
	bool fileIsEmpty;

	//存放往届记录的容器
	map<int, vector<string>>m_Record;

	//成员属性
	//保存第一轮比赛选手编号的容器
	vector<int> v1;

	//保存第一轮晋级选手编号的容器
	vector<int> v2;

	//保存胜出的前三名选手编号的容器
	vector<int> vVictory;

	//存放编号以及对应具体选手的容器
	map<int, Speaker>m_Speaker;

	//存放比赛轮数
	int m_Index;
};

8.3、speechManager.cpp

#include "SpeechManager.h"

//构造函数
SpeechManager::SpeechManager()
{
	//初始化容器和属性
	this->initSpeech();

	//创建12名选手
	this->createSpeaker();

	//加载往届记录
	this->loadRecord();
}

//菜单功能
void SpeechManager::show_Menu()
{
	cout << "************************************" << endl;
	cout << "********  欢迎参加演讲比赛  ********" << endl;
	cout << "********   1.开始演讲比赛   ********" << endl;
	cout << "********   2.查看往届记录   ********" << endl;
	cout << "********   3.清空比赛记录   ********" << endl;
	cout << "********   0.退出比赛程序   ********" << endl;
	cout << "************************************" << endl;
	cout << endl;
}

//初始化容器和属性
void SpeechManager::initSpeech()
{
	//容器都置空
	this->v1.clear();
	this->v2.clear();
	this->vVictory.clear();
	this->m_Speaker.clear();

	//初始化比赛的轮数
	this->m_Index = 1;

	//将记录的容器也清空
	this->m_Record.clear();
}

//创建12名选手
void SpeechManager::createSpeaker()
{
	string nameSeed = "ABCDEFGHIJKL";
	for (int i = 0; i < nameSeed.size(); i++)
	{
		string name = "选手";
		name += nameSeed[i];

		Speaker sp;
		sp.m_Name = name;
		for (int j = 0; j < 2; j++)
		{
			sp.m_Score[j] = 0;
		}

		//创建选手编号并放入v1容器中
		this->v1.push_back(10001 + i);

		//选手编号以及对应选手放入map容器
		this->m_Speaker.insert(make_pair(i+10001 , sp));
	}
}

//抽签
void SpeechManager::speechDraw()
{
	cout << "第 << " << this->m_Index << " >> 轮比赛选手正在抽签" << endl;
	cout << "-----------------------------" << endl;
	cout << "抽签后演讲顺序如下:" << endl;
	if (this->m_Index == 1)
	{
		//第一轮比赛
		random_shuffle(v1.begin(), v1.end());
		for (vector<int>::iterator it = v1.begin(); it != v1.end(); it++)
		{
			cout << *it << " ";
		}
		cout << endl;
	}
	else
	{
		//第二轮比赛
		random_shuffle(v2.begin(), v2.end());
		for (vector<int>::iterator it = v2.begin(); it != v2.end(); it++)
		{
			cout << *it <<" ";
		}
		cout << endl;
	}
	cout << "-----------------------------" << endl;
	system("pause");
	cout << endl;
}

//比赛
void SpeechManager::speechContest()
{
	cout << "----------第 << " << this->m_Index << " >> 轮比赛正式开始!----------" << endl;

	//准备临时容器存放小组成绩
	multimap<double, int, greater<double>> groupScore;
	int num = 0;//用于记录人员个数 - 6人一组

	vector<int>v_Src;//比赛选手容器
	if (this->m_Index == 1)
	{
		v_Src = v1;
	}
	else
	{
		v_Src = v2;
	}

	//遍历所有选手进行比赛
	for (vector<int>::iterator it = v_Src.begin(); it != v_Src.end(); it++)
	{
		num++;

		//评委打分
		deque<double>d;
		for (int i = 0; i < 10; i++)
		{
			double score = (rand() % 401 + 600)/10.0f;//(600 ~ 1000) - (60 ~ 100)
			//cout << score << " ";
			d.push_back(score);
		}
		//cout << endl;

		sort(d.begin(), d.end(), greater<double>());//排序
		d.pop_front();//去除最高分
		d.pop_back();//去除最低分
		double sum = accumulate(d.begin(), d.end(), 0.0f);//总分
		double avg = sum / (double)d.size();//平均分

		//将平均分放入map容器中
		this->m_Speaker[*it].m_Score[this->m_Index - 1] = avg;

		//将打分的数据放入到临时小组容器中
		groupScore.insert(make_pair(avg, *it));//key是得分,value是具体选手的编号
		//每6人取出前三名
		if (num % 6 == 0)
		{
			cout << "第" << num / 6 << "小组比赛名次:"<< endl;
			for (multimap<double, int, greater<double>>::iterator it = groupScore.begin(); it != groupScore.end(); it++)
			{
				cout << "编号:" << it->second << " 姓名:" << this->m_Speaker[it->second].m_Name <<
					" 成绩:" << this->m_Speaker[it->second].m_Score[this->m_Index - 1] << endl;
			}

			//取走前三名
			int count = 0;
			for (multimap<double, int, greater<double>>::iterator it = groupScore.begin(); it != groupScore.end() && count < 3; it++ , count++)
			{
				if (this->m_Index == 1)
				{
					v2.push_back((*it).second);
				}
				else
				{
					vVictory.push_back((*it).second);
				}
			}
			groupScore.clear();//小组容器清空
			cout << endl;
		}
	}
	cout << "----------第 << " << this->m_Index << " >> 轮比赛结束!----------" << endl;
	system("pause");
}

//显示得分
void SpeechManager::showScore()
{
	cout << "----------第 << " << this->m_Index << " >> 轮晋级选手信息如下:----------" << endl;

	vector<int>v;
	if (this->m_Index == 1)
	{
		v = v2;
	}
	else
	{
		v = vVictory;
	}

	for (vector<int>::iterator it = v.begin(); it != v.end(); it++)
	{
		cout << "选手编号:" << *it << " 姓名:" << this->m_Speaker[*it].m_Name << " 成绩:"
			<< this->m_Speaker[*it].m_Score[this->m_Index - 1] << endl;
	}

	cout << endl;
	system("pause");
	system("cls");
	this->show_Menu();
}

//保存分数记录
void SpeechManager::saveRecord()
{
	ofstream ofs;
	ofs.open("speech.csv", ios::out | ios::app);//用追加的方式写文件

	//将每个选手的数据写入文件中
	for (vector<int>::iterator it = vVictory.begin(); it != vVictory.end(); it++)
	{
		ofs << *it << "," << this->m_Speaker[*it].m_Score[1] << ",";
	}
	ofs << endl;

	//关闭文件
	ofs.close();
	cout << "记录已经保存!" << endl;

	//更改文件不为空状态
	this->fileIsEmpty = false;
}

//开始比赛 - 比赛整个流程控制函数
void SpeechManager::startSpeech()
{
	//第一轮开始比赛
	//1.抽签
	this->speechDraw();

	//2.比赛
	this->speechContest();

	//3.显示晋级结果
	this->showScore();

	//第二轮开始比赛
	this->m_Index++;
	//1.抽签
	this->speechDraw();

	//2.比赛
	this->speechContest();

	//3.显示最终结果
	this->showScore();

	//4.保存分数到文件中
	this->saveRecord();

	//重置比赛并获取记录
	//初始化容器和属性
	this->initSpeech();

	//创建12名选手
	this->createSpeaker();

	//加载往届记录
	this->loadRecord();

	cout << "本届大赛举办完毕!" << endl;
	system("pause");
	system("cls");
}

//读取记录
void SpeechManager::loadRecord()
{
	ifstream ifs("speech.csv", ios::in);//读文件
	if (!ifs.is_open())
	{
		this->fileIsEmpty = true;
		ifs.close();
		return;
	}

	//文件清空情况
	char ch;
	ifs >> ch;
	if (ifs.eof())
	{
		this->fileIsEmpty = true;
		ifs.close();
		return;
	}

	//文件不为空
	this->fileIsEmpty = false;
	ifs.putback(ch);//将上面读取的单个字符在放回文件

	int index = 0;
	string data;

	while (ifs >> data)
	{
		vector<string>v;//存放6个string字符串

		int pos = -1;//查到","位置的变量
		int start = 0;

		while (true)
		{
			pos = data.find(",", start);
			if (pos == -1)
			{
				//未找到情况
				break;
			}
			string temp = data.substr(start, pos - start);
			v.push_back(temp);
			start = pos + 1;
		}
		this->m_Record.insert(make_pair(index, v));
		index++;
	}
	ifs.close();
}

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

//清空文件
void SpeechManager::clearRecord()
{
	cout << "是否确认清空文件(Y/N):";
	char select = 0;
	cin >> select;

	if (select == 'Y' || select == 'y')
	{
		//确认清空
		ofstream ofs("speech.csv", ios::trunc);
		ofs.close();

		//初始化容器和属性
		this->initSpeech();

		//创建12名选手
		this->createSpeaker();

		//加载往届记录
		this->loadRecord();

		cout << "清空成功!" << endl;
	}
	system("pause");
	system("cls");
}

//退出系统
void SpeechManager::exitSystem()
{
	cout << "欢迎下次使用!" << endl;
	system("pause");
	exit(0);
}

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

}

 8.4、test.cpp

using namespace std;
#include <iostream>
#include"speechManager.h"

int main()
{
	//添加随机数种子
	srand((unsigned int)time(NULL));

	//创建管理类对象
	SpeechManager sm;

	int choice = 0;//用于存储用户输入
	while (true)
	{
		sm.show_Menu();

		cout << "请输入您的选择:";
		cin >> choice;

		switch (choice)
		{
		case 1://开始比赛
			sm.startSpeech();
			break;
		case 2://查看往届比赛记录
			sm.showRecord();
			break;
		case 3://清空比赛记录
			sm.clearRecord();
			break;
		case 0://退出系统
			sm.exitSystem();
			break;
		default:
			system("cls");//清屏
			break;
		}
	}

	system("pause");
	return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Unity SRP 管线【第九讲:URP 点光源与聚光灯】

文章目录 CPU数据搜集GPU数据使用光照计算 CPU数据搜集 我们只能支持有限数量的其他灯。并将这些灯光数据&#xff08;位置、颜色、阴影强度、方向光光源、灯光遮蔽Probe、灯光层级Mask&#xff09;发送到GPU以供场景中所有物体渲染使用。 //ForwardLights.cs 额外光源数量与…

HarmonyOS4.0系统性深入开发33相对布局(RelativeContainer)

相对布局&#xff08;RelativeContainer&#xff09; 概述 RelativeContainer为采用相对布局的容器&#xff0c;支持容器内部的子元素设置相对位置关系。子元素支持指定兄弟元素作为锚点&#xff0c;也支持指定父容器作为锚点&#xff0c;基于锚点做相对位置布局。下图是一个…

postgresql|数据库|pg_repack插件的部署和使用

一&#xff0c; 表和索引的膨胀现象 Postgres SQL 实现的MVCC的机制不同于 oracle &#xff0c; mysql innodb 的 undo tablespace 的机制。 表上所用的更新和删除等操作的行为&#xff0c;都不会实际的删除或修改&#xff0c;而是标记为死元祖 &#xff08;dead rows or dead…

强化学习原理python篇08——actor-critic

强化学习原理python篇08——actor-critic 前置知识TD ErrorREINFORCEQACAdvantage actor-critic (A2C) torch实现步骤第一步第二步第三步训练结果 Ref 本章全篇参考赵世钰老师的教材 Mathmatical-Foundation-of-Reinforcement-Learning Actor-Critic Methods 章节&#xff0c;请…

灵伴科技(Rokid)借助 Knative 实现 AI 应用云原生 Serverless 化

作者&#xff1a;朱炜栋、元毅、子白 公司介绍 Rokid 创立于 2014 年&#xff0c;是一家专注于人机交互技术的产品平台公司&#xff0c;2018 年即被评为国家高新技术企业。Rokid 作为行业的探索者、领跑者&#xff0c;目前致力于 AR 眼镜等软硬件产品的研发及以 YodaOS 操作系…

【SpringBoot3】Spring 请求处理流程,自定义返回类型处理(HttpMessageConverter)

一、Spring Boot 请求处理 1、请求处理流程 Spring Boot 的接口请求处理流程主要基于 Spring MVC 架构&#xff0c;以下是详细的请求处理流程&#xff1a; 客户端发送请求&#xff1a;客户端发送HTTP请求到Spring Boot应用的URL。 DispatcherServlet 接收请求&#xff1a;Sp…

《C程序设计》上机实验报告(五)之一维数组二维数组与字符数组

实验内容&#xff1a; 1.运行程序 #include <stdio.h> void main( ) { int i,j,iRow0,iCol0,m; int x[3][4]{{1,11,22,33},{2,28,98,38},{3,85,20,89}}; mx[0][0]; for(i0;i<3;i) for(j0;j<4;j) if (x[i][j]>m) { mx[i][j]; iRowi…

QT学习日记 | 信号与槽

目录 前言 一、初始信号与槽 1、信号与槽的本质 2、信号与槽的使用 3、内置信号、内置槽函数与自定义信号、自定义槽函数 &#xff08;1&#xff09;文档查询 &#xff08;2&#xff09;自定义信号与内置槽函数的使用 4、信号与槽函数关联关系 5、带参数的信号与槽函数…

17- OpenCV:图像矩(Image Moments)和点多边形测试

目录 一、图像矩 1、矩的概念介绍 2、相关的API 3、代码演示 二、点多边形测试 1、概念介绍-点多边形测试 2、cv::pointPolygonTest 3、代码演示 一、图像矩 引言 在数字图像处理、计算机视觉与相关领域中&#xff0c;图像矩(Image moments)是指图像的某些特定像素灰…

如何搭建良好的软件测试环境?有什么作用?

在软件开发过程中&#xff0c;测试环境是非常重要的一环。它为软件测试提供了一个模拟真实生产环境的平台&#xff0c;以确保软件的质量和稳定性。那么如何搭建良好的测试环境呢?测试环境又有什么作用呢?卓码软件测评小编将进行以下解答。 软件测试环境是指在软件测试过程中…

Windows10更新失败 错误 0x80070643、KB5034441的解决方法之二

Windows10更新失败 错误 0x80070643、KB5034441 在知乎Windows10更新失败 错误 0x80070643、KB5034441的原因分析和几个解决方法 - 知乎 参考文章进行操作&#xff0c;更详细信息自己看上面链接。 我电脑的硬盘是mbr格式&#xff0c;而且没有划分恢复分区。 Microsoft Windo…

HDFS HA 之 HA 原理

1 ZKFC解析 HA(High Availability)是HDFS支持的一个重要特性,可以有效解决Active Namenode遇到故障时,将可用的Standby节点变成新的Active状态的问题,使集群能够正常工作。目前支持冷切换和热切换两种方式。冷切换通过手动触发,缺点是不能够及时恢复集群。实际生产中以应用…

【数据结构与算法】之排序系列-20240201

【数据结构与算法】之排序系列-20240201 一、88. 合并两个有序数组二、169. 多数元素三、217. 存在重复元素四、242. 有效的字母异位词五、268. 丢失的数字六、349. 两个数组的交集七、350. 两个数组的交集 II 一、88. 合并两个有序数组 简单 给你两个按 非递减顺序 排列的整数…

Linux网络状态查看与防火墙管理

网络状态查看 netstat [选项] Netstat是一款命令行工具&#xff0c;用于显示Linux系统中网络的状态信息&#xff0c;可以显示网络连接、路由表、连接的数据统计等信息。 使用 选项 -a&#xff1a;显示所有选项&#xff0c;包括监听和未监听的端口。 -t&#xff1a;仅显示tc…

C++(17)——list的模拟实现

前面的文章中&#xff0c;介绍了&#xff0c;的模拟实现&#xff0c;本篇文章将介绍对于的模拟实现。 目录 1. list的基本结构&#xff1a; 2. list功能实现&#xff1a;尾部插入元素&#xff1a; 3. list迭代器的实现&#xff1a; 4. list功能实现&#xff1a;在任意位置前…

从零开始训练 RT-DETR模型最新版本教程说明(包含Mac、Windows、Linux端 )同之前的项目版本代码有区别

从零开始训练 RT-DETR- 最新8.1版本教程说明 本文适用Windows/Linux/Mac:从零开始使用Windows/Linux/Mac训练 RT-DETR 算法项目 《芒果 RT-DETR 目标检测算法 改进》 适用于芒果专栏改进 RT-DETR 算法 文章目录 U版 RT-DETR 算法第一步 配置环境1.1 系列配置1.2 代码执行第…

react-virtualized实现行元素不等高的虚拟列表滚动

前言&#xff1a; 当一个页面中需要接受接口返回的全部数据进行页面渲染时间&#xff0c;如果数据量比较庞大&#xff0c;前端在渲染dom的过程中需要花费时间&#xff0c;造成页面经常出现卡顿现象。 需求&#xff1a;通过虚拟加载&#xff0c;优化页面渲染速度 优点&#xff1…

探索组合模式的魅力:探索树形结构的艺术与科学

设计模式专栏&#xff1a;http://t.csdnimg.cn/nolNS 在面对层次结构和树状数据结构的软件设计任务时&#xff0c;我们如何优雅地处理单个对象与组合对象的一致性问题&#xff1f;组合模式&#xff08;Composite Pattern&#xff09;为此提供了一种简洁高效的解决方案。通过本…

网络协议 UDP协议

网络协议 UDP协议 在之前的文章中有对UDP协议套接字的使用进行讲解&#xff0c;本文主要对UDP协议进行一些理论补充。 文章目录 网络协议 UDP协议1. 概念2. UDP协议格式2.1 数据报长度2.2 校验和/检验和2.2.1 CRC校验2.2.2 MD5算法 1. 概念 UDP&#xff0c;即User Datagram P…

[office] excel2010双向条形图制作 #经验分享#微信

excel2010双向条形图制作 本教程为大家介绍一下excel2010中excel2010双向条形图制作方法。 1.选中工作区域 2.点击插入-->图表,选择条形图 3.为美观可将中间竖线可去掉 4.方法是选中竖线,右击-->删除 5.接下来将图例靠上,选中图例,右击-->设置图例格式-->图例选项…