0

0

Tic-Tac-Toe 游戏中的平局检测机制优化

心靈之曲

心靈之曲

发布时间:2025-11-08 23:20:01

|

950人浏览过

|

来源于php中文网

原创

tic-tac-toe 游戏中的平局检测机制优化

本教程详细探讨了在JavaScript实现的Tic-Tac-Toe游戏中,如何精确地检测游戏是否以平局结束。文章分析了传统平局判断逻辑的常见误区,并提出了一种通过跟踪已填充格子数量的优化方案。通过引入filledFields状态变量并在关键函数中进行更新和判断,确保游戏能在无获胜者且棋盘已满时,准确宣布平局,从而提升游戏逻辑的健壮性。

在开发Tic-Tac-Toe(井字棋)这类棋盘游戏时,实现获胜条件检测相对直观,但准确判断游戏是否以平局结束,往往是开发者容易混淆的地方。一个常见的错误是过早地声明平局,即在棋盘尚未完全填满但当前没有获胜者时就判断为平局。本教程将深入探讨Tic-Tac-Toe游戏中平局检测的正确实现方法,并提供一个基于JavaScript的优化方案。

理解游戏状态与获胜条件

在Tic-Tac-Toe游戏中,我们需要管理以下核心状态:

  • board: 一个表示棋盘的数组,通常包含9个元素,每个元素代表一个格子。0表示空,1表示玩家1(X),-1表示玩家2(O)。
  • turn: 当前轮到哪个玩家,通常是1或-1。
  • winner: 游戏结果。null表示游戏进行中,1或-1表示对应玩家获胜,'T'表示平局。
  • COMBOS: 一个包含所有获胜组合的数组,例如[[0, 1, 2], [3, 4, 5], ...]。

getWinner() 函数的核心职责是检查当前棋盘状态,判断是否有玩家获胜,或者游戏是否已经平局。

初始平局检测逻辑的挑战

在原始的getWinner()实现中,开发者可能尝试在遍历所有获胜组合后,如果没有找到赢家,就直接返回'T'。

function getWinner() {
  for (let i = 0; i < COMBOS.length; i++) {
    if (Math.abs(board[COMBOS[i][0]] + board[COMBOS[i][1]] + board[COMBOS[i][2]]) === 3) {
      return board[COMBOS[i][0]]; // 找到赢家
    }
    // 错误示范:此处不应立即检查平局或继续游戏
    // else if (board.includes(null)) { // 原始代码中此处逻辑有误,board初始化为0而非null
    //   return null; // 游戏继续
    // }
  }
  // 错误示范:如果在此处直接返回 'T',游戏会在第一步后就结束
  // return 'T';
}

上述代码的问题在于:

  1. board.includes(null) 的误用:原始代码中board数组初始化为0s,因此board.includes(null)永远不会为真。正确检查空位应使用board.includes(0)。
  2. 过早的平局判断:如果在for循环结束后直接return 'T',意味着只要当前没有立即获胜的组合,游戏就会被判定为平局,即使棋盘上还有很多空位。这会导致游戏在第一步点击后就立即结束。
  3. else if 在循环内的逻辑问题:将else if (board.includes(null))放在循环内部会导致逻辑混乱。它可能在检查完第一个获胜组合后,如果没赢且有空位,就立即返回null,而没有检查其他获胜组合。

优化方案:引入已填充格子计数器

为了准确判断平局,我们需要确保两个条件同时满足:

  1. 没有玩家获胜。
  2. 棋盘上所有格子都已被填充。

我们可以通过引入一个额外的状态变量filledFields来跟踪已填充的格子数量。

1. 声明 filledFields 状态变量

首先,在状态变量区声明filledFields:

viable
viable

基于GPT-4的AI非结构化数据分析平台

下载
/*----- state variables -----*/
let board; // array of 9 boxes
let turn; // 1 or -1
let winner; // null = no winner; 1 or -1 winner; 'T' = Tie
let filledFields; // Counter for filled boxes

2. 初始化 filledFields

在游戏初始化函数init()中,将filledFields设置为0:

/*----- functions -----*/
init();
// Initializes state and calls render()
function init() {
  board = [0, 0, 0, 0, 0, 0, 0, 0, 0];
  turn = 1;
  winner = null;
  filledFields = 0; // 初始化已填充格子计数器
  render();
}

3. 更新 filledFields

在处理玩家点击的handleClick()函数中,每次玩家成功落子后,递增filledFields:

// Get index of the clicked box
function handleClick(event) {
  const boxIdx = parseInt(event.target.id.replace('box-', ''));
  // if statement in case someone clicks outside box, the box is filled or there is a winner
  if (isNaN(boxIdx) || board[boxIdx] || winner)
    return;
  // update state of board with the current turn value
  board[boxIdx] = turn;
  filledFields++; // 有效落子后,递增计数器
  // switch player turn
  turn *= -1;
  // check for a winner
  winner = getWinner();
  render();
}

4. 改进 getWinner() 函数

现在,getWinner()函数可以按照以下逻辑进行重构:

  • 首先,检查是否有玩家获胜。 遍历所有获胜组合,如果找到,则立即返回获胜玩家。
  • 如果没有任何玩家获胜,则检查是否已达到平局条件。 这意味着棋盘上的所有格子都已被填充(即filledFields达到9)。
  • 如果既没有获胜者,也不是平局,则游戏继续。
// Check for a winner in the state. 1(X) or -1(O), 'T' for Tie, null for no winner yet
function getWinner() {
  // 1. 检查是否有玩家获胜
  for (let i = 0; i < COMBOS.length; i++) {
    const [a, b, c] = COMBOS[i];
    if (Math.abs(board[a] + board[b] + board[c]) === 3) {
      return board[a]; // 找到赢家,返回赢家标识
    }
  }

  // 2. 如果没有赢家,检查是否平局 (所有格子都已填充)
  if (filledFields === 9) {
    return 'T'; // 棋盘已满且无赢家,判定为平局
  }

  // 3. 既没有赢家,也不是平局,游戏继续
  return null;
}

渲染消息的调整

renderMessage() 函数已经能够正确处理winner为'T'的情况,因此无需修改:

// Display whose turn it is and the winner
function renderMessage() {
  if (winner === 'T') {
    message.innerHTML = 'Tie Game! Game Over!';
  } else if (winner) {
    message.innerHTML = `Player ${MARK[winner]} Wins!`;
  } else {
    message.innerHTML = `Player ${MARK[turn]}'s Turn`;
  }
}

总结与注意事项

通过引入filledFields计数器,我们能够精确地管理Tic-Tac-Toe游戏的平局检测逻辑。这种方法将平局判断的条件明确为“无获胜者”和“棋盘已满”,避免了在游戏进行中误判平局的问题。

关键要点:

  • 状态管理: 确保filledFields作为游戏状态的一部分,并在init()中初始化,在handleClick()中更新。
  • 逻辑顺序: 在getWinner()函数中,务必先检查获胜条件,再检查平局条件,最后才是游戏继续的条件。
  • 避免冗余检查: 一旦找到赢家,立即返回;一旦判定平局,立即返回。避免不必要的后续计算。
  • 代码清晰性: 保持函数职责单一,getWinner只负责判断游戏结果,handleClick负责处理玩家交互和状态更新。

这种优化方案不仅解决了Tic-Tac-Toe游戏中的平局检测难题,也为其他类似棋盘游戏的逻辑实现提供了宝贵的思路。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

536

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

372

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

706

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

470

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

989

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

652

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

535

2023.09.20

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 2.9万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 1.7万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号