Заполнение логического 2D-массива из текстового файла - PullRequest
0 голосов
/ 04 сентября 2018

У меня возникли небольшие проблемы при попытке ввода данных из текстового файла в двухмерный массив. Данные, которые я пытаюсь импортировать, в основном представляют собой последовательность из 1 и 0, но также с парой символов (S и G). Цель этого упражнения - создать игру-лабиринт, в которой 0 и 1 указывают, заблокировано ли пространство, или нет, а S обозначает начало, а G - цель. Я попробовал несколько вещей, но, похоже, ничего не импортирует должным образом мой текстовый файл для создания первоначального макета лабиринта. Основные ошибки, которые я получаю, это либо массив вне границ, либо он говорит мне, что лабиринт отформатирован неправильно.

Это код для всего класса

package solution;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.Scanner;

/**
 * A maze game.
 * 
 * @author Nicholas Thomas
 * @version 8/30/2018
 *
 */
public class MazeGame
{
    /**
     * The size of each side of the game map.
     */
    private final static int HEIGHT = 19;
    private final static int WIDTH = 39;

    /**
     * The game map, as a 2D array of ints.
     */
    private boolean[][] blocked = new boolean[HEIGHT][WIDTH];

    /**
     * The current location of the player vertically.
     */
    // TODO: add field here.
    int userCol;
    /**
     * The current location of the player horizontally.
     */
    // TODO: add field here.
    int userRow;
    /**
     * The scanner from which each move is read.
     */
    // TODO: add field here.
    Scanner moveScanner = new Scanner(System.in);
    /**
     * The row and column of the goal.
     */
    // TODO: add fields here.

    /**
     * The row and column of the start.
     * 
     * @return
     */
    // TODO: add fields here.

    /**
     * Accessor method for the Blocked array.
     * 
     * @param Blocked
     * 
     * @return Blocked
     * 
     */
    public boolean[][] getBlocked()
    {
        return blocked;
    }

    /**
     * Accessor method to return the user col.
     * 
     * @return userCol
     */
    public int getUserCol()
    {
        return userCol;
    }

    /**
     * Accessor to get userRow.
     * 
     * @return userRow
     */
    public int getUserRow()
    {
        return userRow;
    }

    /**
     * Accessor method to get scanner contents.
     * 
     */
    public Scanner getMoveScanner()
    {
        return moveScanner;
    }

    /**
     * Mutator method for the blocked array.
     * 
     */
    public void setBlocked(boolean[][] blocked)
    {
        this.blocked = blocked;
    }

    /**
     * Mutator method for the userCol.
     */
    public void setUserCol(int col)
    {
        this.userCol = col;
    }

    public void setUserRow(int row)
    {
        this.userRow = row;

    }

    public void setMoveScanner(Scanner moveScanner)
    {
        this.moveScanner = moveScanner;
    }

    /**
     * Constructor initializes the maze with the data in 'mazeFile'.
     * 
     * @param mazeFile
     *            the input file for the maze
     * @throws FileNotFoundException
     */
    public MazeGame(String mazeFile)
    {
        loadMaze(mazeFile);
        moveScanner = new Scanner(System.in);

    }

    /**
     * Constructor initializes the maze with the 'mazeFile' and the move scanner
     * with 'moveScanner'.
     * 
     * @param mazeFile
     *            the input file for the maze
     * @param moveScanner
     *            the scanner object from which to read user moves
     */
    public MazeGame(String mazeFile, Scanner moveScanner)
    {
        loadMaze(mazeFile);
        this.moveScanner = moveScanner;
    }

    /**
     * getMaze returns a copy of the current maze for testing purposes.
     * 
     * @return the grid
     */
    public boolean[][] getMaze()
    {
        if (blocked == null)
        {
            return null;
        }
        boolean[][] copy = new boolean[HEIGHT][WIDTH];
        for (int i = 0; i < HEIGHT; i++)
        {
            for (int j = 0; j < WIDTH; j++)
            {
                copy[i][j] = blocked[i][j];
            }
        }
        return copy;
    }

    /**
     * setMaze sets the current map for testing purposes.
     * 
     * @param maze
     *            another maze.
     */
    public void setMaze(boolean[][] maze)
    {
        this.blocked = maze;
    }

    /**
     * Function loads the data from the maze file and creates the 'blocked' 2D
     * array.
     * 
     * @param mazeFile
     *            the input maze file.
     */
    // TODO: private void loadMaze(String mazeFile)

    /**
     * Actually plays the game.
     */
    public void playGame()
    {

    }

    /**
     * Checks to see if the player has won the game.
     * 
     * @return true if the player has won.
     */
    // TODO: public boolean playerAtGoal()

    /**
     * Makes a move based on the String.
     * 
     * @param move
     *            the direction to make a move in.
     * @return whether the move was valid.
     */
    public boolean makeMove(String move)
    {
        // TODO
        return false;
    }

    /**
     * Prints the map of the maze.
     */
    public void printMaze()
    {
        // TODO
    }

    /**
     * Creates a new game, using a command line argument file name, if one is
     * provided.
     * 
     * @param args
     *            the command line arguments
     */

    public static void main(String[] args)
    {
        String mapFile = "data/easy.txt";
        Scanner scan = new Scanner(System.in);
        MazeGame game = new MazeGame(mapFile, scan);
        game.playGame();
    }

    /**
     * Method to crate the maze from file.
     * 
     * @param mazeFile
     *
     */
    public void loadMaze(String mazeFile)
    {
        Scanner scanner = null;
        try
        {
            String var;
            scanner = new Scanner(new File(mazeFile));
            boolean[][] blocked = new boolean[HEIGHT][WIDTH];
            for (int i = 0; i < HEIGHT; i++)
            {
                for (int j = 0; j < WIDTH; j++)
                {
                    if (scanner.hasNext())
                    {
                        var =scanner.next();

                        if (var == "1")
                        {
                            blocked[i][j] = true;
                        }
                        if (var == "0")
                        {
                            blocked[i][j] = false;
                        }
                        if (var == "S")
                        {
                            blocked[i][j] = false;
                        }
                        if (var == "G")
                        {
                            blocked[i][j] = false;
                        }
                        System.out.print(blocked[i][j] + ",");
                    }
                }
            }
            scanner.close();
        }
        catch (FileNotFoundException e)
        {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

    }
}

Вот как выглядит текстовый файл, который я пытаюсь импортировать.

S 1 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 
0 1 1 1 0 1 1 1 0 1 1 1 0 1 0 1 1 1 0 1 0 1 1 1 1 1 1 1 0 1 0 1 1 1 1 1 0 1 0 
0 0 0 1 0 0 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 0 0 1 0 0 0 1 0 1 0 1 0 0 0 0 0 1 0 
1 1 0 1 1 1 1 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 0 
0 1 0 0 0 1 0 0 0 1 0 1 0 0 0 1 0 1 0 0 0 0 0 1 0 1 0 0 0 1 0 1 0 1 0 0 0 1 0 
0 1 1 1 0 1 0 1 1 1 0 1 1 1 1 1 0 1 1 1 1 1 1 1 0 1 1 1 1 1 1 1 0 1 0 1 1 1 0 
0 0 0 1 0 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 
1 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 0 1 1 1 1 1 0 1 0 1 1 1 0 1 1 1 1 
0 0 0 1 0 1 0 1 0 1 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 1 0 1 0 0 0 0 0 0 0 
0 1 1 1 0 1 1 1 0 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 1 1 0 1 0 1 0 1 1 1 1 1 1 1 0 
0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 1 0 0 0 1 0 0 0 1 G 
0 1 0 1 1 1 0 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 1 1 0 1 1 1 0 1 0 1 0 1 0 1 0 1 1 
0 1 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 1 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 0 0 
0 1 0 1 0 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 0 
0 1 0 0 0 1 0 0 0 1 0 1 0 0 0 0 0 0 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 0 0 0 0 1 0 
0 1 1 1 1 1 0 1 1 1 0 1 1 1 1 1 1 1 0 1 0 1 1 1 0 1 0 1 0 1 0 1 0 1 1 1 0 1 0 
0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 0 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 0 0 
1 1 1 1 0 1 1 1 0 1 0 1 1 1 0 1 0 1 1 1 1 1 0 1 1 1 1 1 1 1 0 1 1 1 0 1 1 1 0 
0 0 0 0 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 

Это первое, что я пытался заполнить массив, но безуспешно.

 /**
     * Method to crate the maze from file.
     * 
     * @param mazeFile
     */
    public void loadMaze(String mazeFile)
    {
        String fileName = ("C:/users/thomasns/easy.txt");
        // File file = new File(fileName);
        Scanner scanMan = new Scanner(fileName);
        boolean blocked[][] = new boolean[HEIGHT][WIDTH];
        while (scanMan.hasNext())
        {
            for (int i = 0; i < blocked.length; i++)
            {
                for (int j = 0; j < blocked[i].length; j++)
                {
                    blocked[HEIGHT][WIDTH] = scanMan.nextBoolean();
                }
                scanMan.close();
            }

        }
    }

Это еще одна реализация, которую я пробовал, на данный момент я могу только заставить это вернуть false для всего.

  /**
     * Method to crate the maze from file.
     * 
     * @param mazeFile
     *
     */
    public void loadMaze(String mazeFile)
    {
        Scanner scanner = null;
        try
        {
            String var;
            scanner = new Scanner(new File(mazeFile));
            boolean[][] blocked = new boolean[HEIGHT][WIDTH];
            for (int i = 0; i < HEIGHT; i++)
            {
                for (int j = 0; j < WIDTH; j++)
                {
                    if (scanner.hasNext())
                    {
                        var =scanner.next();

                        if (var == "1")
                        {
                            blocked[i][j] = true;
                        }
                        if (var == "0")
                        {
                            blocked[i][j] = false;
                        }
                        if (var == "S")
                        {
                            blocked[i][j] = false;
                        }
                        if (var == "G")
                        {
                            blocked[i][j] = false;
                        }
                        System.out.print(blocked[i][j] + ",");
                    }
                }
            }
            scanner.close();
        }
        catch (FileNotFoundException e)
        {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

    }

Если у кого-нибудь есть какой-нибудь совет для меня, я буду очень признателен, заранее спасибо.

1 Ответ

0 голосов
/ 04 сентября 2018

Ваш первый метод может сработать, но вы неправильно указали свои индексы, поскольку они постоянны

Застраховано

 blocked[HEIGHT][WIDTH] = scanMan.nextBoolean();

Вы должны скорее

 blocked[j][i] = scanMan.nextBoolean(); // or other way around.

Второе может сработать, но это не то, как вы сравниваете строки, поэтому застраховано на

 if (var == "1")

должно быть

if (var.equals("1"))

Я либо прочитал символ за символом и поместил его в массив, либо прочитал всю строку, разделил, создал строку массива, повторил.

Первый метод может быть примерно таким:

  boolean blocked[][] = new boolean[HEIGHT][WIDTH];
    int c=0;
    while (scanMan.hasNext()) {
       blocked[c/width][c%width]=scanMan.nextBoolean();
       c++;
    }

я думаю, вам придется обрабатывать другие символы, кроме 1 и 0, но вы должны получить грубое представление.

Рабочий пример:

 public static void main(String... args) {
        String input =
                "S 1 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 \n" +
                "0 1 1 1 0 1 1 1 0 1 1 1 0 1 0 1 1 1 0 1 0 1 1 1 1 1 1 1 0 1 0 1 1 1 1 1 0 1 0 \n" +
                "0 0 0 1 0 0 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 0 0 1 0 0 0 1 0 1 0 1 0 0 0 0 0 1 0 \n" +
                "1 1 0 1 1 1 1 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 0 \n" +
                "0 1 0 0 0 1 0 0 0 1 0 1 0 0 0 1 0 1 0 0 0 0 0 1 0 1 0 0 0 1 0 1 0 1 0 0 0 1 0 \n" +
                "0 1 1 1 0 1 0 1 1 1 0 1 1 1 1 1 0 1 1 1 1 1 1 1 0 1 1 1 1 1 1 1 0 1 0 1 1 1 0 \n" +
                "0 0 0 1 0 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 \n" +
                "1 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 0 1 1 1 1 1 0 1 0 1 1 1 0 1 1 1 1 \n" +
                "0 0 0 1 0 1 0 1 0 1 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 1 0 1 0 0 0 0 0 0 0 \n" +
                "0 1 1 1 0 1 1 1 0 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 1 1 0 1 0 1 0 1 1 1 1 1 1 1 0 \n" +
                "0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 1 0 0 0 1 0 0 0 1 G \n" +
                "0 1 0 1 1 1 0 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 1 1 0 1 1 1 0 1 0 1 0 1 0 1 0 1 1 \n" +
                "0 1 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 1 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 0 0 \n" +
                "0 1 0 1 0 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 0 \n" +
                "0 1 0 0 0 1 0 0 0 1 0 1 0 0 0 0 0 0 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 0 0 0 0 1 0 \n" +
                "0 1 1 1 1 1 0 1 1 1 0 1 1 1 1 1 1 1 0 1 0 1 1 1 0 1 0 1 0 1 0 1 0 1 1 1 0 1 0 \n" +
                "0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 0 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 0 0 \n" +
                "1 1 1 1 0 1 1 1 0 1 0 1 1 1 0 1 0 1 1 1 1 1 0 1 1 1 1 1 1 1 0 1 1 1 0 1 1 1 0 \n" +
                "0 0 0 0 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 ";

        final int width = 39;
        final int heigh = 19;

        Scanner scan = new Scanner(input);
        boolean[][] blocked = new boolean[heigh][width];

        int c = 0;
        while (scan.hasNext()) {
            String character = scan.next();
            switch (character) {
                case "S"://do something; break;
                    System.out.println("Detected S");
                    break;
                case "G"://so something;
                    System.out.println("Detected G");
                    break;
                default:
                    blocked[c / width][c % width] = Boolean.valueOf(character);
            }
            c++;
        }

}
...