news 2026/4/27 5:00:58

UVa 141 The Spot Game

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
UVa 141 The Spot Game

题目分析

The Spot Game\texttt{The Spot Game}The Spot Game是一个基于N×NN \times NN×N棋盘的游戏,双方轮流执行操作,操作包括:

  • 在空白格子中放置一个黑子(用 “+” 表示);
  • 从棋盘上移除一个已有的黑子(用 “-” 表示)。

游戏规则如下:

  • 如果当前棋盘状态(或其旋转90°90°90°180°180°180°后的状态)在之前的游戏过程中已经出现过,则当前玩家立即输掉游戏,对方获胜;
  • 游戏最多进行2N2N2N步;
  • 如果在2N2N2N步后仍未出现重复状态,则游戏平局。

输入包含多组游戏,每组游戏的第一行为棋盘大小NNN2≤N≤502 \leq N \leq 502N50),接下来是2N2N2N行操作,每行包含坐标(x,y)(x, y)(x,y)和操作符+-。输入以000结束。

输出应指出获胜的玩家和获胜的步数,或声明游戏平局。

解题思路

核心问题

我们需要高效地检测棋盘状态是否重复(包括旋转后的状态)。由于棋盘最大可达50×5050 \times 5050×50,直接表示状态矩阵并比较的复杂度较高,但我们可以将状态编码成字符串进行存储和比较。

状态编码

我们可以用一个长度为N×NN \times NN×N的字符串表示棋盘状态:

  • ‘1’表示该位置有黑子;
  • ‘0’表示该位置为空。

每次操作后,更新对应的字符。

旋转处理

题目要求判断当前状态是否与之前任一状态在旋转0°0°90°90°90°(顺时针)、90°90°90°(逆时针)或180°180°180°后相同。因此,我们需要编写函数,分别生成当前状态的三种旋转状态。

为了便于比较,我们始终将旋转后的状态也编码为字符串,并存储到集合中。

查找重复

使用哈希表(如map<string, int>)存储所有出现过的状态(包括旋转后的状态),值为首次出现时的步数。

每步操作后,生成当前状态的四个等价状态(原状态 + 三种旋转),检查它们是否已经在哈希表中:

  • 如果存在,则当前玩家输,对方获胜;
  • 否则,将这四个状态插入哈希表,记录当前步数。

时间复杂度

每一步需要生成四个字符串,长度均为N2N^2N2,最多2N2N2N步,因此总时间复杂度约为O(N3)O(N^3)O(N3),在N≤50N \leq 50N50时是可接受的。

代码实现说明

以下代码使用C++\texttt{C++}C++编写,主要步骤包括:

  1. 读取NNN,若为 0 则结束;
  2. 初始化状态字符串和哈希表;
  3. 循环处理2N2N2N步操作:
    • 更新棋盘状态;
    • 检查当前状态是否已出现;
    • 生成旋转状态并插入哈希表;
  4. 根据是否出现重复状态输出结果。

参考代码

// The Spot Game// UVa ID: 141// Verdict: Accepted// Submission Date: 2016-01-20// UVa Run Time: 0.003s//// 版权所有(C)2016,邱秋。metaphysis # yeah dot net#include<bits/stdc++.h>usingnamespacestd;stringrotateCW90(string matrix,intn){string newMatrix;for(inti=1;i<=n;i++)for(intj=1;j<=n;j++)newMatrix+=matrix[(n-1)*n+(i-1)-(j-1)*n];returnnewMatrix;}stringrotateCCW90(string matrix,intn){string newMatrix;for(inti=1;i<=n;i++)for(intj=1;j<=n;j++)newMatrix+=matrix[(n-1)-(i-1)+(j-1)*n];returnnewMatrix;}stringrotate180(string matrix){reverse(matrix.begin(),matrix.end());returnmatrix;}intmain(){intn,x,y;string line;map<string,int>steps;while(cin>>n,n){intwinner=0,move=0;string matrix=string(n*n,'0');steps.clear();for(inti=1;i<=2*n;i++){cin>>x>>y;getline(cin,line);if(winner>0)continue;matrix[(x-1)*n+(y-1)]=(line.find('+')!=line.npos)?'1':'0';if(steps.find(matrix)!=steps.end()){winner=(steps[matrix]%2==1)?2:1;move=i;}string newMatrix[4]={matrix,rotateCW90(matrix,n),rotateCCW90(matrix,n),rotate180(matrix)};for(intj=0;j<4;j++)if(steps.find(newMatrix[j])==steps.end())steps.insert(make_pair(newMatrix[j],i));}if(winner==0)cout<<"Draw\n";elsecout<<"Player "<<winner<<" wins on move "<<move<<'\n';}return0;}
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 16:22:35

一道“fork + 短路求值”经典题:到底会创建多少个进程?

问题描述 代码如下&#xff08;不算 main 进程本身&#xff0c;问总共创建了多少个子进程&#xff09;&#xff1a; int main(int argc, char* argv[]) {fork();fork() && fork() || fork();fork(); }选项&#xff1a;A.18 B.19 C.20 D.21先把结论放前面 程序最终一…

作者头像 李华
网站建设 2026/4/26 0:10:55

Vite + Vue3 + TS 封装阿里图标 SVG 全局组件

在 Vite Vue3 TS 项目中&#xff0c;封装阿里图标&#xff08;Iconfont&#xff09;为全局 SVG 组件的最佳实践是使用 vite-plugin-svg-icons 插件。这种方式可以将本地下载的 SVG 图标自动打包成 SVG 雪碧图&#xff08;Sprite&#xff09;&#xff0c;方便维护且性能优异。…

作者头像 李华
网站建设 2026/4/26 15:57:55

社会网络仿真软件:NetLogo_(2).NetLogo基础操作

NetLogo基础操作 在这一节中&#xff0c;我们将详细介绍NetLogo的基础操作&#xff0c;包括如何安装和启动NetLogo&#xff0c;如何创建和编辑模型&#xff0c;以及如何运行和观察仿真结果。这些基础操作是使用NetLogo进行社会网络仿真的前提&#xff0c;掌握这些操作将帮助您更…

作者头像 李华
网站建设 2026/4/26 22:01:34

【Python】基础语法入门:顺序、条件与循环

文章目录 一、顺序语句&#xff1a; 从上到下&#xff0c;依次执行二、条件语句&#xff1a;做选择1. 条件语句的三种形式&#xff08;1&#xff09;单条件判断&#xff1a;if语句&#xff08;2&#xff09;双条件判断&#xff1a;if-else语句&#xff08;3&#xff09;多条件判…

作者头像 李华
网站建设 2026/4/20 10:42:55

python---哈夫曼树

关键特性 哈夫曼节点类&#xff08;HuffmanNode&#xff09; 存储字符和频率 支持堆排序的比较操作 哈夫曼树类&#xff08;HuffmanTree&#xff09; 从文本或频率字典构建 自动生成最优编码 支持编码和解码操作 核心功能 build_from_text(): 从文本构建哈夫曼树 encod…

作者头像 李华