Класс не возвращает правильную строку вывода - PullRequest
0 голосов
/ 08 февраля 2020

Я создал класс под названием HangmanManager, который управляет «злой» версией Hangman, где программа не выбирает слово в начале игры. Вместо этого он ждет, пока пользователь угадает символ, с помощью которого затем сужает список слов, содержащих буквы, которые он угадывает.

HangmanMain. java

import java.util.*;
import java.io.*;

public class HangmanMain  {
    public static final String DICTIONARY_FILE = "dictionary2.txt";
    public static final boolean SHOW_COUNT = true;  // show # of choices left

    public static void main(String[] args) throws FileNotFoundException {
        System.out.println("Welcome to the cse143 hangman game.");
        System.out.println();

        // open the dictionary file and read dictionary into an ArrayList
        Scanner input = new Scanner(new File(DICTIONARY_FILE));
        List<String> dictionary = new ArrayList<>();
        while (input.hasNext()) {
            dictionary.add(input.next().toLowerCase());
        }

        // set basic parameters
        Scanner console = new Scanner(System.in);
        System.out.print("What length word do you want to use? ");
        int length = console.nextInt();
        System.out.print("How many wrong answers allowed? ");
        int max = console.nextInt();
        System.out.println();

        // set up the HangmanManager and start the game
        List<String> dictionary2 = Collections.unmodifiableList(dictionary);
        HangmanManager hangman = new HangmanManager(dictionary2, length, max);
        if (hangman.words().isEmpty()) {
            System.out.println("No words of that length in the dictionary.");
        } else {
            playGame(console, hangman);
            showResults(hangman);
        }
    }

    // Plays one game with the user
    public static void playGame(Scanner console, HangmanManager hangman) {
        while (hangman.guessesLeft() > 0 && hangman.pattern().contains("-")) {
            System.out.println("guesses : " + hangman.guessesLeft());
            if (SHOW_COUNT) {
                System.out.println("words   : " + hangman.words().size());
            }
            System.out.println("guessed : " + hangman.guesses());
            System.out.println("current : " + hangman.pattern());
            System.out.print("Your guess? ");
            char ch = console.next().toLowerCase().charAt(0);
            if (hangman.guesses().contains(ch)) {
                System.out.println("You already guessed that");
            } else {
                int count = hangman.record(ch);
                if (count == 0) {
                    System.out.println("Sorry, there are no " + ch + "'s");
                } else if (count == 1) {
                    System.out.println("Yes, there is one " + ch);
                } else {
                    System.out.println("Yes, there are " + count + " " + ch +
                                       "'s");
                }
            }
            System.out.println();
        }
    }

    // reports the results of the game, including showing the answer
    public static void showResults(HangmanManager hangman) {
        // if the game is over, the answer is the first word in the list
        // of words, so we use an iterator to get it
        String answer = hangman.words().iterator().next();
        System.out.println("answer = " + answer);
        if (hangman.guessesLeft() > 0) {
            System.out.println("You beat me");
        } else {
            System.out.println("Sorry, you lose");
        }
    }
}

HangmanManager

import java.util.*;

public class HangmanManager {
   private Set<String> words;
   private Set<Character> guesses;
   private int guessesLeft;
   private String currentPattern;

   // Pre: Throws IllegalArgumentException if length of word is less than 1 or 
   //      if the maximum number of guesses is less than 0
   // Post: Initializes state of the hangman game by taking in the set of all words
   //       from the given dictionary, length of words, and max number of guesses
   public HangmanManager(Collection<String> dictionary, int length, int max) {
      if (length < 1 || max < 0) {
         throw new IllegalArgumentException();
      }
      currentPattern = "";
      guessesLeft = max;
      words = new TreeSet<String>();
      guesses = new TreeSet<Character>();    

      for (String str : dictionary) {
         if (str.length() == length) {
            words.add(str);
         }
      }

      for (int i = 0; i < length; i++) {
         currentPattern += "-";
      }
   }

   // Returns the current set of words being used by the manager
   public Set<String> words() {
      return words;
   }

   // Returns the number of guesses left
   public int guessesLeft() {
      return guessesLeft;
   }

   // Returns the set of characters already guessed by user
   public Set<Character> guesses() {
      return guesses;
   }

   // Pre: Throws IllegalStateException if the set of words is empty
   // Post: Returns the current pattern
   public String pattern() {
      if (words.isEmpty()) {
         throw new IllegalStateException();
      }

      return currentPattern;
   }

   // Pre: Throws IllegalStateException if there are no guesses left or if set of words is empty
   // Pre: Throws IllegalArgumentException if the user guess was already guessed before
   // Post: Updates the dictionary by determining which set of words from it to choose from in
   //       response to user guesses. Returns occurrences of guessed letters in the new pattern.
   public int record(char guess) {
      if(guessesLeft < 1 || words.isEmpty()) {
         throw new IllegalStateException();
      }

      if (guesses.contains(guess)) {
         throw new IllegalArgumentException();
      }

      int count = 0;
      Map<String, Set<String>> mapPattern = new TreeMap<String, Set<String>>();
      for (String mapWords : words) {
         String newPattern = constructPattern(guess, mapWords);
         if (mapPattern.containsKey(newPattern)) {
            mapPattern.get(newPattern).add(mapWords);
         } else {
            Set<String> setPattern = new TreeSet<String>();
               setPattern.add(mapWords);
               mapPattern.put(newPattern, setPattern);
         }
      }
      String key = getLargestSet(mapPattern);
      if (currentPattern.equals(key)) {
         guessesLeft--;
      }
      currentPattern = key;
      words = mapPattern.get(key);
      guesses.add(guess);
      return count;        
   }

   private String getLargestSet(Map<String, Set<String>> map) {
      int max = 0;
      String pattern = "";
      for (String key : map.keySet()) {
         if (map.get(key).size() > max) {
            max = map.get(key).size();
            pattern = key;
         }
      }
      return pattern;
   }



   private String constructPattern(char guess, String word) {
      String pattern = "";
      for (int i = 0; i < word.length(); i++) {
         if (guess == word.charAt(i)) {
            pattern += Character.toString(guess);
         } else {
            pattern += currentPattern.substring(i, i + 1);
         }
      }
      return pattern;
   }
}

Это один пример вывода того, что я хочу отобразить

Welcome to the cse143 hangman game.

What length word do you want to use? 4
How many wrong answers allowed? 7

guesses : 7
words   : 9
guessed : []
current : - - - -
Your guess? e
Sorry, there are no e's

guesses : 6
words   : 3
guessed : [e]
current : - - - -
Your guess? o
Yes, there are 2 o's

guesses : 6
words   : 2
guessed : [e, o]
current : - o o -
Your guess? t
Sorry, there are no t's

guesses : 5
words   : 2
guessed : [e, o, t]
current : - o o -
Your guess? f
Sorry, there are no f's

guesses : 4
words   : 2
guessed : [e, f, o, t]
current : - o o -
Your guess? c
Sorry, there are no c's

guesses : 3
words   : 1
guessed : [c, e, f, o, t]
current : - o o -
Your guess? n
Sorry, there are no n's

guesses : 2
words   : 1
guessed : [c, e, f, n, o, t]
current : - o o -
Your guess? d
Yes, there is one d

guesses : 2
words   : 1
guessed : [c, d, e, f, n, o, t]
current : - o o d
Your guess? h
Sorry, there are no h's

guesses : 1
words   : 1
guessed : [c, d, e, f, h, n, o, t]
current : - o o d
Your guess? g
Yes, there is one g

answer = good
You beat me

Это текущий вывод из моей программы при запуске HangmanMain

Welcome to the cse143 hangman game.

What length word do you want to use? 4
How many wrong answers allowed? 7

guesses : 7
words   : 9
guessed : []
current : ----
Your guess? e
Sorry, there are no e's

guesses : 6
words   : 3
guessed : [e]
current : ----
Your guess? o
Sorry, there are no o's

guesses : 6
words   : 2
guessed : [e, o]
current : -oo-
Your guess? t
Sorry, there are no t's

guesses : 5
words   : 2
guessed : [e, o, t]
current : -oo-
Your guess? f
Sorry, there are no f's

guesses : 4
words   : 2
guessed : [e, f, o, t]
current : -oo-
Your guess? c
Sorry, there are no c's

guesses : 3
words   : 1
guessed : [c, e, f, o, t]
current : -oo-
Your guess? n
Sorry, there are no n's

guesses : 2
words   : 1
guessed : [c, e, f, n, o, t]
current : -oo-
Your guess? d
Sorry, there are no d's

guesses : 2
words   : 1
guessed : [c, d, e, f, n, o, t]
current : -ood
Your guess? h
Sorry, there are no h's

guesses : 1
words   : 1
guessed : [c, d, e, f, h, n, o, t]
current : -ood
Your guess? g
Sorry, there are no g's

answer = good
You beat me

Несмотря на то, что шаблон current обновляется в ответ на догадки пользователя, программа постоянно утверждает, что в его «выбранном» слове нет букв.

...