Минимаксная функция не работает для подключения 4

Я пытаюсь реализовать компьютерную версию AI Connect 4 в Java, используя минимаксный алгоритм. Для этой задачи я создал класс "Узел", который начинается с одного состояния платы, а затем рекурсивно создает больше узлов до заданной глубины. Затем он оценивает каждый узел с 1 как победа, 0 как ничья или максимальная глубина. Наконец, я использую минимаксный алгоритм, чтобы выбрать, какой шаг использовать в функции "getColumn()". Вот мой код

package ai;

import java.util.ArrayList;
import java.util.Arrays;

public class Node {
    public ArrayList < Node > nodeList = new ArrayList < Node > ();
    private int[][] board; // Represents board, 0 is empty space, 1 is player 1, 2 is computer
    private int turn; // Current turn, 1 is player, 2 is computer

    // Node constructor, creates tree of possible game states
    public Node(int[][] board, int depth, int turn) {
        this.board = board;
        this.turn = turn;

        if (depth > 0 && result(turn) == -1) {
            int[][] copy = new int[7][6];

            for (int i = 0; i < 7; i++)
            copy[i] = board[i].clone();

            int[] cols = getAvailCols(board);

            for (int col: cols) {
                int y = placeBlock(col, turn, copy);
                nodeList.add(new Node(copy, depth - 1, turn == 1 ? 2 : 1));
                copy[col][y] = 0;
            }
        }
    }

    // Returns 0 for draw, 1 for win, -1 for no result
    private int result(int currTurn) {
        int horCount = 1;
        int verCount = 1;
        int ldiaCount = 1;
        int rdiaCount = 1;

        for (int x = 0; x < 7; x++) {
            for (int y = 0; y < 6; y++) {
                for (int countX = x + 1; countX < 7; countX++) {
                    if (board[countX][y] == currTurn) horCount++;
                    else break;
                }

                for (int countX = x - 1; countX >= 0; countX--) {
                    if (board[countX][y] == currTurn) horCount++;
                    else break;
                }

                for (int countY = y + 1; countY < 6; countY++) {
                    if (board[x][countY] == currTurn) verCount++;
                    else break;
                }

                for (int countY = y - 1; countY >= 0; countY--) {
                    if (board[x][countY] == currTurn) verCount++;
                    else break;
                }

                for (int countX = x + 1, countY = y + 1; countX < 7 && countY < 6; countX++, countY++) {
                    if (board[countX][countY] == currTurn) rdiaCount++;
                    else break;
                }

                for (int countX = x - 1, countY = y - 1; countX >= 0 && countY >= 0; countX--, countY--) {
                    if (board[countX][countY] == currTurn) rdiaCount++;
                    else break;
                }

                for (int countX = x + 1, countY = y - 1; countX < 7 && countY >= 0; countX++, countY--) {
                    if (board[countX][countY] == currTurn) ldiaCount++;
                    else break;
                }

                for (int countX = x - 1, countY = y + 1; countX >= 0 && countY < 6; countX--, countY++) {
                    if (board[countX][countY] == currTurn) ldiaCount++;
                    else break;
                }

                if (horCount >= 4 || verCount >= 4 || ldiaCount >= 4 || rdiaCount >= 4) {
                    return 1;
                }

                for (int i = 0; i < 7; i++) {
                    for (int j = 0; j < 6; j++) {
                        if (board[i][j] == 0) return -1;
                    }
                }
            }
        }

        return 0;
    }

    // Puts chip into a column
    private int placeBlock(int x, int turn, int[][] checkboard) {
        int y = 0;

        while (y < 5 && checkboard[x][y + 1] == 0)
        y++;

        checkboard[x][y] = turn;
        return y;
    }

    // Gets the score of a node
    private int score() {
        int result = result(turn);

        if (result == 0) return 0;
        else if (result == 1) return Integer.MAX_VALUE;
        else if (nodeList.size() > 0) {
            int score = Integer.MIN_VALUE;

            for (Node node: nodeList)
            score = Math.max(score, -node.score());

            return score;
        } else {
            return 0;
        }
    }

    public int[] getAvailCols(int[][] checkboard) {
        ArrayList < Integer > list = new ArrayList < Integer > ();

        for (int i = 0; i < 7; i++) {
            if (checkboard[i][0] == 0) list.add(i);
        }

        int[] cols = new int[list.size()];

        for (int i = 0; i < list.size(); i++) {
            cols[i] = (int) list.get(i);
        }

        return cols;
    }

    // Only called for top node tell whichmove ai should take
    public int getColumn() {
        int[] cols = getAvailCols(board);
        int index = 0;
        int maxScore = Integer.MIN_VALUE;

        for (int i = 0; i < cols.length; i++) {
            if (nodeList.get(i).score() > maxScore) {
                maxScore = nodeList.get(i).score();
                index = i;
            }
        }

        return cols[index];
    }
}

В массиве платы:

  • 0с пустое пространство
  • 1s фишки игроков
  • 2s фишки ай

Однако я провел много тестов, и компьютер всегда выбирает самый левый столбец, который доступен, даже когда другой игрок собирается выиграть. Может ли кто-нибудь объяснить, как это исправить или в чем проблема. Я сейчас использую глубину 7 для своей игры.

0 ответов

Другие вопросы по тегам