以游戏编程的角度看待模拟时间的算法题——以PAT甲级1026 Table Tennis为例

对于需要模拟时间的算法题,可以将开始时间作为游戏的开始(如Unity的Start或UE的BeginPlay),每一秒的模拟作为游戏的画面更新(如Unity的Update或UE的Tick),结束时间可作为游戏的结束(如Unity的OnDestroy或UE的EndPlay)。

从这个角度来看,我们可以将PAT甲级1026 Table Tennis当成一个游戏看待,在游戏进行过程中,游戏玩家可能随时会让一位球员来打球。因此可以将可能到来的球员当成玩家输入来处理。

在每个时刻(每一秒),我们读取玩家输入,然后按照指定的游戏玩法(题目的规则)来进行游戏数据的更新。

对于游戏玩法,其要点如下:

  • 球员可能是VIP球员也可能是一个普通球员;
  • VIP球员唯一享有的特权就是当有空闲的VIP球桌,他可以上VIP球桌打球。(这意味着:如果没有空闲的VIP球桌,VIP球员就是普通球员;如果没有VIP球员在等待打球,那么VIP球桌就是普通球桌。……看似很复杂,其实不影响游戏逻辑。)

因此按照第二点玩法我们可以给出游戏在每一秒的逻辑:

  1. 获取玩家输入,获得该时刻到来的球员(题目说每秒只会到来一个球员),将其加入球员等待队列,对于VIP球员,将其额外加入VIP球员等待队列。
  2. 遍历每一个VIP球桌,按照VIP球员到来的顺序让其上球桌打球
  3. 遍历每一个球桌,按照球员到来的顺序让其上球桌打球
#include<bits/stdc++.h>
using namespace std;const int kOpenTime  = 8 * 60 * 60;  // 开门时间 8点
const int kCloseTime = 21 * 60 * 60; // 关门时间 21点// 玩家类
struct Player {int  arrvTime; // 球员到达时间int  playTime; // 球员占用球桌的时间int  waitTime; // 球员等待时间bool isVip;// 球员是不是vipint  num;      // 球员的输出的编号static const int kInvalidNum = 0xffffff;Player(): arrvTime(kOpenTime), playTime(0), waitTime(0), isVip(false), num(kInvalidNum) {}[[nodiscard]] bool isPlayed() const { return num !=  kInvalidNum; };
};// 球桌类
struct Table {int  endTime;  // 上一对球员打完球离开该球桌的时间int  id;bool isVip;// 是不是vip桌int  serveNum; // 桌子招待过多少球员Table(): id{-1}, endTime(kOpenTime), isVip(false), serveNum(0) {}[[nodiscard]] bool isFree(int nowTime) const { return endTime <= nowTime; }
};// 游戏场景中某个对象所挂载的脚本类
class TableTennisManager
{
private:vector<Player>  m_players;// 下面两个queue内存储的是m_players内元素的指针queue<Player *> playersWaitQueue;queue<Player *> vipPlayersWaitQueue;int m_unarrivedIDBegin = 0;int m_playedCnt = 0; // 记录球员上桌的顺序vector<Table *> m_tables_VIP;vector<Table *> m_tables;// 将每个时刻到来的球员当做一个输入,将该球员放入队列中让HandleQueue来处理void HandleInput(int nowTime) {for(int playerID = m_unarrivedIDBegin; playerID < m_players.size(); playerID++) {Player & player = m_players[playerID];if(player.arrvTime <= nowTime) {m_unarrivedIDBegin++;playersWaitQueue.emplace(&player);if(player.isVip)vipPlayersWaitQueue.emplace(&player);} else {break;}}}void HandleQueue(int nowTime, vector<Table *> & tables, queue<Player *> & waitPlayers){// 清除已处理过的球员while(!waitPlayers.empty()) {if(waitPlayers.front()->isPlayed()) {waitPlayers.pop();} else {break;}}for(auto table: tables) {if(!table->isFree(nowTime))continue;if(waitPlayers.empty())break;/* 该桌空闲,让正在等待的球员上桌 */// 弹出等待队列,处理该球员Player * player = waitPlayers.front();waitPlayers.pop();// 更新球员数据player->waitTime = nowTime - player->arrvTime;player->num = m_playedCnt++;// 更新球桌数据table->endTime = nowTime + player->playTime;table->serveNum++;}}public:/// 游戏开始,接收游戏过程中可能需要的数据输入,模拟玩家输入并构造游戏场景————在第一次调用Update前调用// m_players模拟玩家输入,m_tables是游戏场景所需的游戏对象void Begin(){int nPlayers, nTables, nVipTables; // 球员对的数量、球桌数、VIP球桌数// 输入在营业时间内到达的球员数据cin >> nPlayers;m_players.resize(nPlayers);int nPlayers_real = 0; // 记录符合要求的输入数据(到达时间在21点之前的球员)for(int i = 0; i < nPlayers; ++i) {// 输入:每队球员的到达时间、占用球桌的时间(最多不能超过两小时)、是否是VIPint h, m, s, playT, isVip;scanf("%d:%d:%d %d %d\n", &h, &m, &s, &playT, &isVip); //NOLINTint arvT = (h*60 + m)*60 + s; // 按照秒数记录player到达时间// 只保留在关门前到达的球员if(arvT < kCloseTime){Player player;player.arrvTime = arvT;player.playTime = playT > 120 ? 120 * 60 : playT * 60; // 打球超过2小时的都要强制变成120分钟player.isVip = isVip;m_players[nPlayers_real++] = player;}}m_players.resize(nPlayers_real);// 输入球桌数据cin >> nTables >> nVipTables;m_tables.resize(nTables);for(int i = 0; i < nTables ; i++) {m_tables[i] = new Table{};}// 输入VIP球桌数据for(int i = 0; i < nVipTables; ++i) {int tableID;cin >> tableID;tableID--;m_tables[tableID]->isVip = true; // 标记vip桌m_tables[tableID]->id = tableID;m_tables_VIP.push_back(m_tables[tableID]);}// 按照球员到达时间排序sort(m_players.begin(), m_players.end(), [](Player &a, Player &b){return a.arrvTime < b.arrvTime;});}/// 更新游戏数据,更新游戏场景数据(m_players和m_tables)————每个时刻(每秒)调用一次void Update(int nowTime){// 处理该时刻可能到达的球员HandleInput(nowTime);// 在VIP球桌上处理VIP球员HandleQueue(nowTime, m_tables_VIP, vipPlayersWaitQueue);// 在所有球桌上处理所有球员(上一步因为VIP球桌已满而未能进入VIP球桌的VIP球员将会被视为普通球员处理)HandleQueue(nowTime, m_tables, playersWaitQueue);}/// 游戏结束,输出游戏执行结果————在最后一次Update执行完调用void End(){// 将球员按照打球的顺序进行排序sort(m_players.begin(), m_players.end(), [](Player &a, Player &b){return a.num < b.num;});// 输出球员的打球信息for(Player &player: m_players) {if(player.num == 0xffffff) continue; // 超过21点才能打球的球员就不能输出了int ah = player.arrvTime/3600, am = player.arrvTime % 3600 / 60, as = player.arrvTime % 60;int bh = (player.arrvTime + player.waitTime) / 3600, bm = (player.arrvTime + player.waitTime) % 3600 / 60, bs = (player.arrvTime + player.waitTime) % 60;// 等待的分钟数要四舍五入player.waitTime = player.waitTime/60 + (player.waitTime%60 >= 30);printf("%02d:%02d:%02d %02d:%02d:%02d %d\n",ah,am,as,bh,bm,bs,player.waitTime);}// 输出球桌的接待信息for(int i = 0; i < m_tables.size(); ++i) {cout << m_tables[i]->serveNum << (i == m_tables.size() - 1 ? "\n" : " ");delete m_tables[i];}}
};int main() {TableTennisManager tennisManager;tennisManager.Begin();for (int nowTime = kOpenTime; nowTime < kCloseTime; nowTime++) {// 每秒调用一次,对相关数据进行更新tennisManager.Update(nowTime);}tennisManager.End();return 0;
}

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

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

相关文章

Django使用uwsgi+nginx部署,admin没有样式解决办法

Django使用uwsginginx部署,admin没有样式解决办法 如果使用了虚拟环境则修改nginx.conf文件中的/static/路径为你虚拟环境的路径&#xff0c;没有使用虚拟环境则改为你python安装路径下的static server {listen 8008;server_name location; #改为自己的域名&#xff0c;没域名…

【Linux】五种IO模型

文章目录 1. IO基本概念2. 五种IO模型2.1 五个钓鱼的例子2.2 五种IO模型2.2.1 阻塞IO2.2.2 非阻塞IO2.2.3 信号驱动IO2.2.4 IO多路转接2.2.5 异步IO 1. IO基本概念 认识IO IO就是输入和输出&#xff0c;在冯诺依曼体系结构中&#xff0c;将数据从输入设备拷贝到内存就叫输入&am…

力扣热题100 JavaScript--136. 只出现一次的数字

给你一个 非空 整数数组 nums &#xff0c;除了某个元素只出现一次以外&#xff0c;其余每个元素均出现两次。找出那个只出现了一次的元素。 你必须设计并实现线性时间复杂度的算法来解决此问题&#xff0c;且该算法只使用常量额外空间。 示例 1 &#xff1a; 输入&#xff…

【Linux】信号捕捉

目录 信号捕捉1.用户态与内核态1.1关于内核空间与内核态&#xff1a;1.2关于用户态与内核态的表征&#xff1a; 2.信号捕捉过程 信号捕捉 1.用户态与内核态 用户态&#xff1a;执行用户代码时&#xff0c;进程的状态 内核态&#xff1a;执行OS代码时&#xff0c;进程的状态 …

中国1km分辨率月最低温度数据集(1901-2021年)介绍

该数据为中国逐月最低温度数据&#xff0c;空间分辨率为0.0083333&#xff08;约1km&#xff09;&#xff0c;时间为1901.1-2021.12。数据格式为NETCDF&#xff0c;即.nc格式。数据单位为0.1 ℃。该数据集是根据CRU发布的全球0.5气候数据集以及WorldClim发布的全球高分辨率气候…

Mac强制停止应用

有时候使用Mac的时候&#xff0c;某个应用卡住了&#xff0c;但是肯定不能因为一个应用卡住了&#xff0c; 就将电脑重启吧&#xff0c;所以只需要单独停止该应用即可&#xff0c;使用快捷键optioncommandesc就会出现强制停止的界面&#xff0c;选择所要停止的应用&#xff0c;…

第一课-前提-Stable Diffusion 教程

学习 SD 的前提是电脑配置! SD 参考配置: 建议选择台式机 i5 CPU, 内存16GB,N卡 RTX3060, 8G显存以上的配置(最低配) 在此基础上的配置越高越好。 比如,cpu i7 更好,显卡能有 RTX4090 更好,32显存要能有最好,嘿嘿嘿。 如何查看自己的显卡配置? Win+R 输入 “dxdiag…

从零开始,探索Python变量的奥秘!

在Python这个充满魔力的编程世界中&#xff0c;变量是一门必须深入了解的基础知识。无论你是初学者还是有经验的开发者&#xff0c;掌握变量的基本语法对于编写优雅、高效的代码至关重要。本篇博客将带你从零开始&#xff0c;深入探索Python变量的奥秘&#xff0c;让你在编程之…

wait,notify/notifyAll都要放在synchronized锁里面

wait&#xff0c;notify/notifyAll都要放在synchronized锁里面 如果没放在synchronized锁里面&#xff0c;就要报这样的错误 public class Test5 {public static void main(String[] args) throws InterruptedException {Thread t1 new Thread(()->{syn();},"t1&quo…

在消费级GPU调试LLM的三种方法:梯度检查点,LoRA和量化

LLM的问题就是权重参数太大&#xff0c;无法在我们本地消费级GPU上进行调试&#xff0c;所以我们将介绍3种在训练过程中减少内存消耗&#xff0c;节省大量时间的方法:梯度检查点&#xff0c;LoRA和量化。 梯度检查点 梯度检查点是一种在神经网络训练过程中使动态计算只存储最…

26 MFC序列化函数

文章目录 Serialize对于存储文件的序列化 Serialize Serialize 是一个在 MFC (Microsoft Foundation Classes) 中常用的函数或概念。它用于将对象的数据进行序列化和反序列化&#xff0c;便于在不同的场景中保存、传输和恢复对象的状态。 在 MFC 中&#xff0c;Serialize 函数…

EdgeBox_tx1_A200 PyTorch v1.9.0 环境部署

大家好&#xff0c;我是虎哥&#xff0c;今天远程帮助几个小伙伴在A200 控制器上安装PyTorch v1.9.0 torchvision v0.10.0&#xff0c;中间也是经历了很多波折&#xff0c;当然&#xff0c;大部分是网络问题和版本适配问题&#xff0c;所以完事后&#xff0c;将自己完整可用的过…

esp32c3 xiao 脚本记录

oled显示网络时间, wifi链接网络 // ntp_get_date.h #include "time.h"String week[8] {"Sun", "Mon", "Tues", "Wednes", "Thur", "Fri", "Sat" };void printLocalTime(Adafruit_SSD1306 …

开源进展 | WeBASE v3.1.0发布,新增多个实用特性

WeBASE是一个友好、功能丰富的区块链中间件平台&#xff0c;通过一系列通用功能组件和实用工具&#xff0c;助力社区开发者更快捷地与区块链进行交互。 目前WeBASE已更新迭代至v3.1.0版本&#xff0c;本次更新中&#xff0c;WeBASE带来了最新的合约Java脚手架导出功能&#xff…

vue3 实现一个下拉刷新

1. 实现最简单的下拉刷新雏形 <template><div class"wrap" ref"freshcontainer" touchstart"handlerstart" touchmove"handlermove" touchend"handlerend"><div class"fresh_txt" v-if"mo…

【音视频SDK测评】线上K歌软件开发技术选型

摘要 在线K歌软件的开发有许多技术难点&#xff0c;需考虑到音频录制和处理、实时音频传输和同步、音频压缩和解压缩、设备兼容性问题等技术难点外&#xff0c;此外&#xff0c;开发者还应关注音乐版权问题&#xff0c;确保开发的应用合规合法。 前言 前面写了几期关于直播 …

vue项目在body设置公共的背景前提下,区分首页背景图和其他页面背景图

1.需求:在vue项目已设置统一的body背景图的前提,单独给首页换一个背景图,然后其他页面背景图不变的临时需求 实现思路1:在首页home.vue中 在公共的style.css文件中写上两个背景样式(写在公共样式中是因为style.css比组件内部的先加载,避免页面出现后背景空白的问题) …

十四.redis哨兵模式

redis哨兵模式 1.概述2.测试3.哨兵模式优缺点 redis哨兵模式基础是主从复制 1.概述 主从切换的技术方法&#xff1a;当主节点服务器宕机后&#xff0c;需要手动把一台从服务器切换为主服务器&#xff0c;这就需要人工干预&#xff0c;费时费力&#xff0c;还会造成一段时间内服…

单例模式和工厂模式

目录 今日良言&#xff1a;关关难过关关过&#xff0c;步步难行步步行 一、单例模式 1.饿汉模式 2.懒汉模式 二、工厂模式 今日良言&#xff1a;关关难过关关过&#xff0c;步步难行步步行 一、单例模式 首先来解释一下&#xff0c;什么是单例模式。 单例模式也就是单个…

【动态规划刷题 4】礼物的最大价值下降路径最小和

礼物的最大价值 在一个 m*n 的棋盘的每一格都放有一个礼物&#xff0c;每个礼物都有一定的价值&#xff08;价值大于 0&#xff09;。你可以从棋盘的左上角开始拿格子里的礼物&#xff0c;并每次向右或者向下移动一格、直到到达棋盘的右下角。给定一个棋盘及其上面的礼物的价值…