首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Python中最简单的Tic Tac Toe AI

Python中最简单的Tic Tac Toe AI
EN

Code Review用户
提问于 2021-01-18 06:51:07
回答 2查看 517关注 0票数 0

上周,我挑战了自己,为一个以人工智能为对手的Tic脚趾游戏创建尽可能小的代码。使用的字符数最少。游戏的要求如下:

  • 一种“不错”的游戏体验(每次移动后都能获得用户的输入和打印板)
  • 处理错误的输入数据而不崩溃。
  • 有一个无敌的人工智能作为对手。
  • 游戏结束后再玩或退出的能力。

其结果是这样的代码:

代码语言:javascript
复制
def p(b):
    c = [' ' if i == 0 else 'X' if i == -1 else 'O' if i == 1 else i for i in b]
    [print(f'\n{c[r*3:3+r*3]}') if r == 0 else print(c[r*3:3+r*3]) for r in range(3)]
def e(b, t):
    for p in ([0, 1, 2], [3, 4, 5], [6, 7, 8], [0, 3, 6], [1, 4, 7], [2, 5, 8], [0, 4, 8], [2, 4, 6]):
        if b[p[0]] == b[p[1]] == b[p[2]] == t: return 1
def n(b, d, t):
    if e(b, t): return 0, (9+d)
    if e(b, -t): return 0, -(9 + d)
    if 0 not in b: return 0, 0
    x = -20
    for m in [i for i in range(9) if not b[i]]:
        b[m] = t
        s = -n(b, d - 1, -t)[1]
        b[m] = 0
        if s > x: x, y = s, m
    return y, x
def r():
    b, w = [0]*9, -1
    p(b)
    while 1:
        if w == -1 and not (e(b, w) or e(b, -w)) and 0 in b:
            while 1:
                u = input('\nPlease enter your move (1-9): ')
                if u.isnumeric():
                    u = int(u)-1
                    if 0 <= u < 9 and not b[u]:
                        b[u], w = -1, w*-1
                        p(b)
                        break
        elif w == 1 and not (e(b, w) or e(b, -w)) and 0 in b:
            m, s = n(b, 8, 1)
            b[m], w = 1, w*-1
            p(b)
        else:
            f = 'You won!' if e(b, -1) else 'AI won!' if e(b, 1) else 'Game drawn!'
            if input(f'\n{f} Do you want to play again (y/n)? ') != 'y': break
            r()
            break
r()

为了更好地理解正在发生的事情,请给出一点评论:

代码语言:javascript
复制
# Printing the board on the "standard" format with X:s and O:s instead of -1, 0, 1. 
def print_board(board):
    new_board = [' ' if i == 0 else 'X' if i == -1 else 'O' if i == 1 else i for i in board]
    [print(f'\n{new_board[row*3:3+row*3]}') if row == 0 else print(new_board[row*3:3+row*3]) for row in range(3)]  # Print with new line to get nicer format
    
# Evaluates the board
def evaluate(board, turn):
    for pos in ([0, 1, 2], [3, 4, 5], [6, 7, 8], [0, 3, 6], [1, 4, 7], [2, 5, 8], [0, 4, 8], [2, 4, 6]):  # Go through all possible winning lines
        if board[pos[0]] == board[pos[1]] == board[pos[2]] == turn: return 1  # Return 1 if player turn has 3 in a row

# Recursive negamax function which goes through the entire game tree.
# Depth d is used in the returned scores to get shortest possible route to victory. 
def negamax(board, depth, turn):
    if evaluate(board, turn): return 0, (9+depth)  # Return positive score if maximizing player
    if evaluate(board, -turn): return 0, -(9 + depth)  # Return negative score if minimizing player wins
    if 0 not in board: return 0, 0  # Drawn game, return 0
    best_score = -20  # Initiate with less than smallest possible score
    for move in [i for i in range(9) if not board[i]]:  # Go through all empty squares on board
        board[move] = turn  # Make move
        score = -negamax(board, depth - 1, -turn)[1]  # Recursive call to go through all child nodes
        board[move] = 0  # Unmake the move
        if score > best_score: best_score, best_move = score, move  # If score is larger than previous best, update score
    return best_move, best_score  # Return the best move and its corresponding score

# Main game loop.
def run():
    board, turn = [0]*9, -1  # Initiate board and turn to move (-1 is human to start, 1 AI to start)
    print_board(board)
    while 1:  # Loop until game is over
        if turn == -1 and not (evaluate(board, turn) or evaluate(board, -turn)) and 0 in board:  # Human turn if game is not over and there are places left on board
            while 1:  # Loop until a valid input is given
                user_input = input('\nPlease enter your move (1-9): ')  # Get input
                if user_input.isnumeric():  # Find if a number is entered
                    u = int(user_input)-1  # Get it on the right board format (square 1 corresponds to array[0])
                    if 0 <= u < 9 and not board[u]:  # Check if number is in the correct range and on an empty square
                        board[u], turn = -1, turn*-1  # Make move and change turn
                        print_board(board)
                        break
        elif turn == 1 and not (evaluate(board, turn) or evaluate(board, -turn)) and 0 in b:  # Ai turn if game is not over and there are places left on board
            move, score = negamax(board, 8, 1)  # Run Negamax loop to get a best move and the score
            board[move], turn = 1, turn*-1  # Make move and change turn
            print_board(board)
        else:  # This means the game is over or board is full
            text = 'You won!' if evaluate(board, -1) else 'AI won!' if evaluate(board, 1) else 'Game drawn!'  # Check who won or if there is a draw
            if input(f'\n{text} Do you want to play again (y/n)? ') != 'y': break  # Ask to play again, break if answer is not 'y'
            run()  # Run game again if answer is 'y'
            break
run()  # Run the game loop

我的问题是,是否还有其他更简单的方法,在一个功能游戏的字符数量与上述给定的要求?当然,控制台的输入/输出文本可以更短,但我认为这不算:)

就可读性和PEP 8风格而言,当然还有很多需要改进的地方,我想将代码保持在合理的最低限度。

我希望这类问题在这里是允许的,否则请删除它。

编辑:用户在评论中提出的“超级雨”游戏示例:

代码语言:javascript
复制
['  ', ' ', ' ']\
[' ', ' ', ' ']\
[' ', ' ', ' ']

Please enter your move (1-9): 5

[' ', ' ', ' ']\
[' ', 'X', ' ']\
[' ', ' ', ' ']

['O', ' ', ' ']\
[' ', 'X', ' ']\
[' ', ' ', ' ']

Please enter your move (1-9): 4

['O', ' ', ' ']\
['X', 'X', ' ']\
[' ', ' ', ' ']

['O', ' ', ' ']\
['X', 'X', 'O']\
[' ', ' ', ' ']

Please enter your move (1-9): 2

['O', 'X', ' ']\
['X', 'X', 'O']\
[' ', ' ', ' ']

['O', 'X', ' ']\
['X', 'X', 'O']\
[' ', 'O', ' ']

Please enter your move (1-9): 3

['O', 'X', 'X']\
['X', 'X', 'O']\
[' ', 'O', ' ']

['O', 'X', 'X']\
['X', 'X', 'O']\
['O', 'O', ' ']

Please enter your move (1-9): 9

['O', 'X', 'X']\
['X', 'X', 'O']\
['O', 'O', 'X']

Game drawn! Do you want to play again (y/n)? 
EN

回答 2

Code Review用户

发布于 2021-01-18 13:14:35

你的

对于pos in (0,1,2,3、4、5,6、7、8,0,3,6,1、4、7,2、5、8,0,4,8,2、4、6):如果板[pos0] ==板[pos1] == board[pos2] ==转弯:>返回1

可能是

代码语言:javascript
复制
def evaluate(board, turn):
    for i, j, k in [0, 1, 2], [3, 4, 5], [6, 7, 8], [0, 3, 6], [1, 4, 7], [2, 5, 8], [0, 4, 8], [2, 4, 6]:
        if board[i] == board[j] == board[k] == turn: return 1

也就是说,对于索引,单个字母ijk是常见的,而且完全可以。元组不需要括号。

您也可以用两个值而不是三个值来定义每一行。就像我不久前写的一样:

代码语言:javascript
复制
def show():
    for i in range(0, 9, 3):
        print(*board[i : i+3])

def possible_moves(board):
    return (i for i in range(9) if isinstance(board[i], int))

def move(board, p, i):
    return board[:i] + [p] + board[i+1:]

def won(board):
    for i, d in (0, 1), (3, 1), (6, 1), (0, 3), (1, 3), (2, 3), (0, 4), (2, 2):
        if board[i] == board[i + d] == board[i + 2*d]:
            return True

def value(board, p, q):
    if won(board):
        return -1
    return -min((value(move(board, p, i), q, p)
                 for i in possible_moves(board)),
                default=0)

def best_move():
    return min(possible_moves(board),
               key=lambda i: value(move(board, 'O', i), 'X', 'O'))

board = list(range(1, 10))
try:
    while True:
        show()
        board = move(board, 'X', int(input('your move: ')) - 1)
        if won(board):
            raise Exception('You won!')
        if board.count('X') == 5:
            raise Exception('Draw!')
        board = move(board, 'O', best_move())
        if won(board):
            raise Exception('You lost!')
except Exception as result:
    show()
    print(result)

演示游戏:

代码语言:javascript
复制
1 2 3
4 5 6
7 8 9
your move: 5
O 2 3
4 X 6
7 8 9
your move: 3
O 2 X
4 X 6
O 8 9
your move: 4
O 2 X
X X O
O 8 9
your move: 8
O O X
X X O
O X 9
your move: 9
O O X
X X O
O X X
Draw!

就像我不久前写的一样,你只是给了我一个理由把它扔在这里:-)。所以我的程序不是对你的程序进行“评论”,它不处理无效的输入,也不要求另一个游戏(用户可以再次运行这个程序),但也许你会在其中找到一些有用的东西。

票数 2
EN

Code Review用户

发布于 2022-10-14 02:29:55

你可以在一个已经满了的正方形上打字。

票数 0
EN
页面原文内容由Code Review提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://codereview.stackexchange.com/questions/254875

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档