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

2次阅读

如何正确实现井字棋(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(),提升可读性与复用性。

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

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

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界面打下坚实基础。

text=ZqhQzanResources