0

0

如何正确实现井字棋(Tic-Tac-Toe)游戏中的平局判定逻辑

霞舞

霞舞

发布时间:2026-02-07 13:38:24

|

902人浏览过

|

来源于php中文网

原创

如何正确实现井字棋(Tic-Tac-Toe)游戏中的平局判定逻辑

本文详解井字棋中平局(draw)逻辑失效的根本原因,并提供精准修复方案:将平局检测移至第5轮玩家1行动后,而非循环计数i==9处,同时给出结构优化与可维护性增强的完整实践指南。

在井字棋这类双人轮流落子的3×3游戏中,平局(draw)发生在全部9个格子填满且无人获胜时。但许多初学者误以为只需在主循环执行到第9次迭代(i == 9)时检查平局——这会导致逻辑失效,因为每轮for循环实际处理两名玩家各一次移动,共2步。因此,当第9个格子被填满时,循环仅执行了5轮(玩家1走第1、3、5、7、9步;玩家2走第2、4、6、8步),i的最大值为5,永远达不到9。

✅ 正确的平局判定位置与条件

应将平局检查放在玩家1完成第5次移动之后、玩家2开始第5次移动之前。此时若仍未分出胜负,说明9格已满,必为平局。修正后的代码片段如下:

for i in range(1, 6):  # 循环5次即可覆盖全部9步(P1:5步,P2:4步)
    # Player 1 move
    valid_move = False
    while not valid_move:
        move = input("Player1 your turn with X. Please choose a cell (1-9): ")
        if not move.isdigit() or not ('1' <= move <= '9'):
            print("Please insert a valid cell number from 1-9.")
        else:
            valid_move = make_move(player1, move)
            if valid_move:
                printBoard(theBoard)

    # ✅ 平局检查:玩家1完成第5步后(即第9格被填满)
    if i == 5:
        if not check_winner(theBoard, player1) and not check_winner(theBoard, player2):
            print("It's a draw!")
            reset_board()
            break

    # 检查玩家1是否获胜
    if check_winner(theBoard, player1):
        print("Player1 wins!")
        reset_board()
        break

    # Player 2 move(仅执行前4轮)
    if i < 5:
        valid_move = False
        while not valid_move:
            move = input("Player2 your turn with O. Please choose a cell (1-9): ")
            if not move.isdigit() or not ('1' <= move <= '9'):
                print("Please insert a valid cell number from 1-9.")
            else:
                valid_move = make_move(player2, move)
                if valid_move:
                    printBoard(theBoard)

        # 检查玩家2是否获胜
        if check_winner(theBoard, player2):
            print("Player2 wins!")
            reset_board()
            break
? 关键改进点:将 range(1, 10) 改为 range(1, 6),语义更清晰(共5轮决策周期);平局判断前置至 i == 5 且双重验证无胜者(避免因获胜检测遗漏导致误判);抽离重复逻辑为函数:check_winner(board, player) 和 reset_board(),提升可读性与复用性。

? 推荐重构:解耦逻辑与展示,消除重复

原始代码中胜负检测、输入处理、重置操作高度耦合且重复出现。专业写法应遵循单一职责原则:

Face++旷视
Face++旷视

Face⁺⁺ AI开放平台

下载
def check_winner(board, player):
    """检查指定玩家是否达成三连"""
    wins = [
        ['7','8','9'], ['4','5','6'], ['1','2','3'],
        ['7','4','1'], ['8','5','2'], ['9','6','3'],
        ['7','5','3'], ['9','5','1']
    ]
    return any(all(board[pos] == player for pos in combo) for combo in wins)

def reset_board():
    global theBoard
    theBoard = {k: ' ' for k in theBoard}

def get_valid_move(player):
    while True:
        move = input(f"Player {player}, enter cell (1-9): ").strip()
        if move.isdigit() and move in theBoard and theBoard[move] == ' ':
            return move
        print("Invalid input — please choose an empty cell from 1 to 9.")

主循环由此大幅简化,聚焦流程控制:

for round_num in range(1, 6):
    # Player X
    move = get_valid_move('X')
    make_move('X', move)
    printBoard(theBoard)
    if check_winner(theBoard, 'X'):
        print("Player X wins!")
        reset_board()
        break
    if round_num == 5:
        print("It's a draw!")
        reset_board()
        break

    # Player O
    move = get_valid_move('O')
    make_move('O', move)
    printBoard(theBoard)
    if check_winner(theBoard, 'O'):
        print("Player O wins!")
        reset_board()
        break

⚠ 注意事项与最佳实践

  • 输入校验必须包含位置有效性:仅判断isdigit()不够,还需确保输入在'1'–'9'范围内且对应格子为空;
  • 避免全局变量滥用:理想情况下,theBoard应作为参数传入函数,而非依赖global;
  • 平局判定需在双方均未获胜前提下触发:直接 i == 5 不足,应补上 not check_winner(...) 双重保障;
  • 边界测试不可少:模拟填满棋盘但无三连的场景(如标准“猫步”布局),验证是否准确输出 It's a draw!。

通过以上修正与重构,你的井字棋不仅能准确识别平局,还将具备良好的扩展性与可维护性——为后续添加AI对手、计分系统或GUI界面打下坚实基础。

相关文章

在线游戏
在线游戏

海量精品小游戏合集,无需安装即点即玩,休闲益智、动作闯关应有尽有,秒开即玩,轻松解压,快乐停不下来

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

84

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

98

2025.09.18

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.02.06

java多线程方法汇总
java多线程方法汇总

本专题整合了java多线程面试题、实现函数、执行并发相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.02.06

1688阿里巴巴货源平台入口与批发采购指南
1688阿里巴巴货源平台入口与批发采购指南

本专题整理了1688阿里巴巴批发进货平台的最新入口地址与在线采购指南,帮助用户快速找到官方网站入口,了解如何进行批发采购、货源选择以及厂家直销等功能,提升采购效率与平台使用体验。

82

2026.02.06

快手网页版入口与电脑端使用指南 快手官方短视频观看入口
快手网页版入口与电脑端使用指南 快手官方短视频观看入口

本专题汇总了快手网页版的最新入口地址和电脑版使用方法,详细提供快手官网直接访问链接、网页端操作教程,以及如何无需下载安装直接观看短视频的方式,帮助用户轻松浏览和观看快手短视频内容。

15

2026.02.06

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

9

2026.02.06

Python 微服务架构与 FastAPI 框架
Python 微服务架构与 FastAPI 框架

本专题系统讲解 Python 微服务架构设计与 FastAPI 框架应用,涵盖 FastAPI 的快速开发、路由与依赖注入、数据模型验证、API 文档自动生成、OAuth2 与 JWT 身份验证、异步支持、部署与扩展等。通过实际案例,帮助学习者掌握 使用 FastAPI 构建高效、可扩展的微服务应用,提高服务响应速度与系统可维护性。

5

2026.02.06

JavaScript 异步编程与事件驱动架构
JavaScript 异步编程与事件驱动架构

本专题深入讲解 JavaScript 异步编程与事件驱动架构,涵盖 Promise、async/await、事件循环机制、回调函数、任务队列与微任务队列、以及如何设计高效的异步应用架构。通过多个实际示例,帮助开发者掌握 如何处理复杂异步操作,并利用事件驱动设计模式构建高效、响应式应用。

7

2026.02.06

热门下载

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

精品课程

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

共21课时 | 3.4万人学习

Git版本控制工具
Git版本控制工具

共8课时 | 1.5万人学习

Git中文开发手册
Git中文开发手册

共0课时 | 0人学习

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

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