基于STL泛化编程的演讲比赛 小案例 c++

简介: 基于STL泛化编程的演讲比赛 小案例 c++

导航


1.总结

2.代码


——————————————————————————————————————


1.总结

花了半天多点时间去看的黑马程序员c++一个基于stl小实例


总结点:

用到了vector,deque,map,multimap容器


1.其中建立三个vector容器主要用于存放比赛人员的编号,第一次比赛,最后一次比赛和胜利的最后三人

2.deque容器用来存放10个评委打出来的分数,为什么用deque呢,因为这个容器是双端容器,对头尾好进行操作

3.map容器用来保存人员的类和他的编号

4.multimap容器主要是存放每个小组6人的排名并且取出来放入下一组比赛之中


大概过程见自己手写纸上


题目的要求:




—————————————————————————————————————使


使用工具:visual stdio 2012

两个头文件,两个源文件



speaker.h


#pragma once 
#include <iostream>
using namespace std;
//创建选手类
class Speaker
{
public:
  string m_name;
  double m_score[2]; //进行两轮比较,最多放两次
};


speechManager.h


#pragma once 
#include <iostream>
using namespace std;
#include <vector>
#include <map>
#include <deque>
#include "speaker.h"
#include <string>
#include <algorithm>
#include <functional>
#include <numeric>   //用于累加算法
#include <fstream>
class speechManager
{
public:
  //构造函数
  speechManager();
  //菜单界面
  void show_Menu();
  //退出系统
  void Exist_System();
  //初始化容器
  void init_speech();
  //创建选手
  void create_speaker();
  //开始比赛
  void start_speech();
  //抽签
  void speakerDraw();
  //比赛函数
  void speechContest();
  //显示得分
  void showScore();
  //保存分数
  void saveRecord();
  //读取记录
  void loadRecord();
  //判断文件是否为空
  bool fileIsEmpyt;
  //存放往届记录的容器
  map<int,vector<string>> m_Record;
  //显示往届记录
  void showRecord();
  //清空记录
  void clearRecord();
  //析构函数
  ~speechManager();
  vector<int> v1; //用于存放初始的12个选手
  vector<int> v2;  //第一轮晋级的选手
  vector<int> vvictory; //最后晋级三位选手
  map<int,Speaker> m_speaker;   //存放编号以及对应选手容器
  int speech_num;  //比赛的轮数
};


speechManager.cpp


#pragma once
#include "speechManager.h"
//构造函数
speechManager::speechManager()
{
  //初始化容器
  this->init_speech();
  //创建选手
  this->create_speaker();
  //读取文件
  this->loadRecord();
}
//菜单界面
void speechManager::show_Menu()
{
  cout<<"*************欢迎使用****************"<<endl;
  cout<<"**********1.开始演讲比赛*************"<<endl;
  cout<<"**********2.查看往届记录*************"<<endl;
  cout<<"**********3.清空比赛记录*************"<<endl;
  cout<<"**********4.退出比赛程序*************"<<endl;
}
//退出系统
void speechManager::Exist_System()
{
  cout<<"欢迎再次使用"<<endl;
  system("pause");
  exit(0);
}
//初始化容器
void speechManager::init_speech()
{
  this->v1.clear();
  this->v2.clear();
  this->vvictory.clear();
  this->m_speaker.clear();
  //初始化轮数
  this->speech_num = 1;
  //清空读取的容器
  this->m_Record.clear();
}
//创建选手
void speechManager::create_speaker()
{
  string name = "ABCDEFGHIJKM";
  //创建12个选手
  for(int i=0;i<name.size();i++)
  {
  string nam = "选手";
  nam += name[i];
  //创建选手
  Speaker sk;
  sk.m_name = nam;
  for(int j=0;j<2;j++)
  {
    sk.m_score[j] = 0;
  }
  //将选手放入容器中
  this->v1.push_back(i+10001);  //存入的是选手编号
  //map容器中存入编号和创建的类
  this->m_speaker.insert(make_pair(i+10001,sk));  
  }
}
//开始比赛
void speechManager::start_speech()
{
  //第一轮比赛
  //抽签
  this->speakerDraw();
  //比赛
  this->speechContest();
  //显示晋级结果
  this->showScore();
  //第二轮比赛
  this->speech_num++;
  //抽签
  this->speakerDraw();
  //比赛
  this->speechContest();
  //显示最后结果
  this->showScore();
  //保存
  this->saveRecord();
  //重置比赛
  //初始化容器,获取记录
  this->init_speech();
  //创建12名选手
  this->create_speaker();
  //加载往届记录
  this->loadRecord();
  cout<<"本届比赛正式结束"<<endl;
  system("pause");
  system("cls");
}
//抽签
void speechManager::speakerDraw()
{
  cout<<"开始抽签"<<endl;
  cout<<"*************************"<<endl;
  cout<<"抽取顺序如下:"<<endl;
  //根据比赛轮数进行打乱
  if(this->speech_num == 1)
  {
  random_shuffle(v1.begin(),v1.end());  //进行打乱
  for(vector<int>::iterator it = v1.begin();it!=v1.end();it++)
  {
    cout<<*it<<" ";
  }
  }
  else
  {
  random_shuffle(v2.begin(),v2.end());  //进行打乱
  for(vector<int>::iterator it = v2.begin();it!=v2.end();it++)
  {
    cout<<*it<<" ";
  }
  }
  cout<<endl<<"*************************"<<endl;
  system("pause");   //进行缓冲一下
}
//比赛函数
void speechManager::speechContest()
{
  cout<<"----------第"<<this->speech_num<<"轮比赛正式开始——————"<<endl;
  //准备临时容器 存放小组成绩
  multimap<double,int,greater<double>> groupScore;
  //用于统计人员个数,6人一组
  int num=0;
  vector<int>v_Scr;  //比赛选手容器  
  //根据比赛的场次来进行容器赋值
  if(this->speech_num == 1)
  {
  v_Scr = v1;
  }
  else
  {
  v_Scr = v2;
  }
  //遍历所有选手进行比赛
  for(vector<int>::iterator it = v_Scr.begin();it!=v_Scr.end();it++)
  {
  num++;
  //评委打分
  deque<double>d;  //对于去掉最高分最低分使用deque更好
  for(int i=0;i<10;i++)
  {
    double score = (rand()%401+600)/10.2f;  //得到的数是小数
    //cout<<score<<" ";
    d.push_back(score);
  }
  //cout<<endl;
  sort(d.begin(),d.end(),greater<double>());  //对double容器进行降序排序
  d.pop_front();  //去掉最高分
  d.pop_back();   //去掉最低分
  double sum = accumulate(d.begin(),d.end(),0.0f);  //进行累加,总分
  double avg = sum/(double)d.size();  //求平均值
  //可以打印以下平均分
  //cout<<"标号为:"<<*it<<" 姓名为:"<<this->m_speaker[*it].m_name<<" 平均分:"<<avg<<endl;
  //将平均分放入map容器中
  this->m_speaker[*it].m_score[this->speech_num-1] = avg;
  //将打分数据,放入到临时容器group中
  groupScore.insert(make_pair(avg,(*it)));   //key是得分,value是具体选手编号
  //每6人取前三名
  if(num%6 == 0)
  {
    cout<<"第"<<num/6<<"小组比赛名次:"<<endl;
    //打印6人中对应的信息
    for(multimap<double,int,greater<double>>::iterator its = groupScore.begin();its!=groupScore.end();its++)
    cout<<"编号:"<<its->second<<" 姓名:"<<this->m_speaker[its->second].m_name<<" 成绩:"<<this->m_speaker[its->second].m_score[this->speech_num-1]<<endl;
    //取掉每组中的前三名
    int count = 0;
    for(multimap<double,int,greater<double>>::iterator its = groupScore.begin();its!=groupScore.end();its++)
    {
    if(count<3)
      if(this->speech_num == 1)  //如果是第一轮放入v2
      {
      v2.push_back(its->second);
      }
      else  //否则放入vvctory
      {
      vvictory.push_back(its->second);
      }
    count++;
    }
    //清空容器
    groupScore.clear();
  }
  }
  cout<<endl;
  cout<<"-------------第"<<this->speech_num<<"轮比赛结束-------------"<<endl;
  system("pause");
}
//显示得分
void speechManager::showScore()
{
  cout<<"-------------第"<<this->speech_num<<"轮晋级选手信息如下-------------"<<endl;
  vector<int> v;  //作为临时容器,好下面直接遍历
  if(this->speech_num == 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->speech_num-1]<<endl;
  }
  system("pause");
} 
//保存分数
void speechManager::saveRecord()
{
  ofstream ofs;
  ofs.open("victoryPer.csv",ios::out|ios::app);
  for(vector<int>::iterator it = vvictory.begin();it!=vvictory.end();it++)
  {
  ofs<<*it<<","<<m_speaker[*it].m_score[1]<<",";
  }
  ofs<<endl;
  ofs.close();
  system("pause");
  system("cls");
  cout<<"记录已保存"<<endl;
  this->fileIsEmpyt = false;
}
//读取记录
void speechManager::loadRecord()
{
  ifstream ifs("victoryPer.csv",ios::in);   //读文件
  if(!ifs.is_open())  //直接判断文件是否存在
  {
  this->fileIsEmpyt = true;
  //cout<<"文件不存在"<<endl;
  ifs.close();
  return;
  }
  char ch;
  ifs>>ch;   //读取一个字符
  if(ifs.eof())  //判断文件为空的情况
  {
  //cout<<"文件为空"<<endl;
  this->fileIsEmpyt = true;
  ifs.close();
  return;
  }
  //文件不为空
  this->fileIsEmpyt = false;
  ifs.putback(ch);  //将上面读取的字符放回来
  string data;
  int index = 0;
  while(ifs>>data)  //读取文件
  {
  vector<string> v;  //存放6个string字符串
  //cout<<data;
  //进行解析
  int start = 0;
  int pos=-1;
  while(true)
  {
    pos = data.find(",",start);  //从start位置开始查找
    if(pos == -1)
    {
    break;
    }
    string str = data.substr(start,pos-start);  //从start开始pos—start个位置
    v.push_back(str);   //将读取到的存入容器
    start = pos + 1;
  }
  this->m_Record.insert(make_pair(index,v));   //存入第几届和对应的容器
  index++;
  }
  ifs.close();
  循环遍历这个map容器
  //for(map<int,vector<string>>::iterator it = m_Record.begin();it!=m_Record.end();it++)
  //{
  //  cout<<"第"<<it->first<<"届冠军编号:"<<it->second[0]<<" 成绩为:"<<it->second[1]\
  //  <<"   季军编号:"<<it->second[2]<<" 成绩为:"<<it->second[3]\
  //  <<"   亚军编号:"<<it->second[4]<<" 成绩为:"<<it->second[5]<<endl;
  //}
}
//显示往届记录
void speechManager::showRecord()
{
  if(this->fileIsEmpyt)
  {
  cout<<"文件为空或不存在"<<endl;
  }
  else
  {
  for(map<int,vector<string>>::iterator it = m_Record.begin();it!=m_Record.end();it++)
  {
    cout<<"第"<<it->first+1<<"界冠军编号:"<<it->second[0]<<" 成绩为:"<<it->second[1]\
    <<"   季军编号:"<<it->second[2]<<" 成绩为:"<<it->second[3]\
    <<"   亚军编号:"<<it->second[4]<<" 成绩为:"<<it->second[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)
  {
  ofstream ofs("victoryPer.csv",ios::trunc);
  ofs.close();
  //初始化容器
  this->init_speech();
  //创建选手
  this->create_speaker();
  //读取文件
  this->loadRecord();
  cout<<"清理成功"<<endl;
  }
  system("pause");
  system("cls");
}
//析构函数
speechManager::~speechManager()
{
}


演讲比赛管理.cpp


#include <iostream>
using namespace std;
#include "speechManager.h"
#include <ctime>
int main()
{
  srand((unsigned int)time(NULL));  //创建随机种子
  //创建管理类
  speechManager m;
  int choice;  //选项
  测试是否创建12个选手,打印出编号
  //for(map<int,Speaker>::iterator it = m.m_speaker.begin();it !=m.m_speaker.end();it++)
  //{
  //  cout<<"选手编号为:"<<it->first<<" 姓名为:"<<it->second.m_name<<" 初始成绩:"<<it->second.m_score[0]<<endl;
  //}
  while(true)
  {
  m.show_Menu();
  cout<<"请输入您想要的选项"<<endl;
  cin>>choice;
  switch(choice)
  {
  case 1:  //开始演讲比赛
    m.start_speech();
    break;
  case 2:  //查看往届记录
    //m.loadRecord();
    m.showRecord();
    break;
  case 3:  //清空比赛记录
    m.clearRecord();
    break;
  case 4:  //退出比赛程序
    m.Exist_System();
    break;
  default:
    system("cls");
    break;
  }
  }
  system("pause");
  return 0;
}


相关文章
|
8天前
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
15 1
|
3月前
|
存储 C++ UED
【实战指南】4步实现C++插件化编程,轻松实现功能定制与扩展
本文介绍了如何通过四步实现C++插件化编程,实现功能定制与扩展。主要内容包括引言、概述、需求分析、设计方案、详细设计、验证和总结。通过动态加载功能模块,实现软件的高度灵活性和可扩展性,支持快速定制和市场变化响应。具体步骤涉及配置文件构建、模块编译、动态库入口实现和主程序加载。验证部分展示了模块加载成功的日志和配置信息。总结中强调了插件化编程的优势及其在多个方面的应用。
431 67
|
21天前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
37 7
|
2月前
|
存储 编译器 C语言
【c++丨STL】vector的使用
本文介绍了C++ STL中的`vector`容器,包括其基本概念、主要接口及其使用方法。`vector`是一种动态数组,能够根据需要自动调整大小,提供了丰富的操作接口,如增删查改等。文章详细解释了`vector`的构造函数、赋值运算符、容量接口、迭代器接口、元素访问接口以及一些常用的增删操作函数。最后,还展示了如何使用`vector`创建字符串数组,体现了`vector`在实际编程中的灵活性和实用性。
69 4
|
2月前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
81 5
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
64 2
|
2月前
|
存储 算法 Linux
【c++】STL简介
本文介绍了C++标准模板库(STL)的基本概念、组成部分及学习方法,强调了STL在提高编程效率和代码复用性方面的重要性。文章详细解析了STL的六大组件:容器、算法、迭代器、仿函数、配接器和空间配置器,并提出了学习STL的三个层次,旨在帮助读者深入理解和掌握STL。
62 0
|
24天前
|
存储 编译器 C语言
【c++丨STL】vector模拟实现
本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
31 0
|
3月前
|
安全 程序员 编译器
【实战经验】17个C++编程常见错误及其解决方案
想必不少程序员都有类似的经历:辛苦敲完项目代码,内心满是对作品品质的自信,然而当静态扫描工具登场时,却揭示出诸多隐藏的警告问题。为了让自己的编程之路更加顺畅,也为了持续精进技艺,我想借此机会汇总分享那些常被我们无意间忽视却又导致警告的编程小细节,以此作为对未来的自我警示和提升。
384 11
|
2月前
|
消息中间件 存储 安全