Juego MasterMind Java necesito ayuda

Iniciado por Rasgaroth, 1 Febrero 2015, 17:24 PM

0 Miembros y 1 Visitante están viendo este tema.

Rasgaroth

Tengo que hacer el juego del MasterMind en java, por el momento he conseguido que me genere una clave aleatoria de 4 colores entre 6 colores existentes dentro del juego que seria lo que habria que adivinar, introduciendo por teclado una secuencia de 4 colores.

El juego trata de introducir una secuencia de colores en un orden concreto y si coincide con la que el programa habia generado en un principio ganas, pero si fallas pueden ocurrir dos cosas, que no aciertes ninguna posicion pero si el color y te apareceria un mensaje que te diria "Hay X fichas descolocadas" y " 0 fichas acertadas" o podria ser que acertases alguna y hubiese alguna descolocada. El problema lo tengo al intentar sacar el metodo de fichasAcertadas y fichasDescolocadas

pongo el codigo que tengo y a ver si algun alma caritativa me pudiese resolver estos dos metodos. Perdon por lo del code, no lo sabia.

Código (java) [Seleccionar]
import java.util.Random;
import java.util.Scanner;

public class MasterMind {

public static void main(String[] args) {
//char [] clave = GenerarClave();
   
System.out.println("Bienvenido al juego del MasterMind");
System.out.println("");
System.out.println("Escribe una secuencia de 4 colores introduciendo la letras del color elegido y pulsando intro hasta completar la secuencia.");
System.out.println("M=Marron, R=Rojo, A=Azul, Z=Amarillo, N=Naranja y V=Verde");
System.out.println("");
MostrarSolucion(pedirSolucion());
}

public static String [] pedirSolucion()  {
String [] solucion = new String [4];
Scanner input = new Scanner(System.in);
System.out.println("Escriba la soluccion que crea que es correcta: ");
for (int i = 0; i < solucion.length; i = i + 1) {
String color = input.next();
solucion[i] = color;
}
return solucion;
}
 
public static void MostrarSolucion(String [] solucion){
for (int i = 0; i < solucion.length; i = i + 1) {
System.out.print(solucion[i] + " ");
}
System.out.println();
}

static boolean estaRepetido(char color,char [] clave){
boolean repetido = false;
int i=0;
while ( (i < clave.length)&&(repetido == false) ){
if (color == clave[i]){
repetido=true;
}
i++;
}
return repetido;
}

public static char[] GenerarClave(){
Random rnd = new Random();
char [] clave = new char[4];
int i=0;
char posibleColor = 'Q';
char posibleColor2 = 'Q';
while(i < clave.length){
if(i==0){
double x = rnd.nextDouble();
int color =(int) ((x*5)+1);
switch (color) {
case 1: clave[i] = 'M';break;
case 2: clave[i] = 'R';break;
case 3: clave[i] = 'N';break;
case 4: clave[i] = 'A';break;
case 5: clave[i] = 'Z';break;
case 6: clave[i] = 'V';break;
}
i++;
}else{
double j = rnd.nextDouble();
int color3 =(int) ((j*5)+1);
switch (color3) {
case 1: posibleColor = 'M';break;
case 2: posibleColor = 'R';break;
case 3: posibleColor = 'N';break;
case 4: posibleColor = 'A';break;
case 5: posibleColor = 'Z';break;
case 6: posibleColor = 'V';break;
}
posibleColor2 = posibleColor;
while (estaRepetido(posibleColor2,clave)){
double y = rnd.nextDouble();
int color2 =(int) ((y*5)+1);
switch (color2) {
case 1: posibleColor2 = 'M';break;
case 2: posibleColor2 = 'R';break;
case 3: posibleColor2 = 'N';break;
case 4: posibleColor2 = 'A';break;
case 5: posibleColor2 = 'Z';break;
case 6: posibleColor2 = 'V';break;
}
}
clave[i] = posibleColor2;
i++;
}
}
return clave;
}
}



Mod: Tema movido a Java, procura usar los Sub-foros correctos

3n31ch

#1
Amigo, si no pones las etiquetas code no se entiende mucho por favor. Es una regla del foro y tiene sus razones. Modifica tu mensaje y selecciona el código y fíjate que hay un combobox que dice GeShi seleccionas Java y ya esta. Todos felices.

Cuando hagas eso intentare ver tu código porque la verdad es que así como esta me da pereza.




Modifico. Gracias por agregar las etiquetas code, leí las reglas así que ahora voy a intentar ayudarte (intentare hacerlo sin código por ahora, pero si realmente te ves superado puedo intentar ayudarte un poco mas. Pero recuerda que estos tipos de ejercicios necesitan que tu emplees la lógica y así aprendes.

Primero que todo esos System.out.println("") para hacer una linea en blanco no gustan nada, mejor pon un \n:
System.out.println("Bienvenido al juego del MasterMind\n"); // El \n es un salto de linea

Existen mas secuencias de escape como estas: \b, \t, \r, \n, \', \", \\. Puedes ver que hace cada una aquí

Tambien en el ciclo for de pedir solucion, te recomiendo remplazar el i = i + 1 por un i++ (que quiere decir que le sumas uno) en caso de sumar de dos en dos puedes utilizar i+=2 que es lo mismo que i = i + 2. Si quieres mas info de esto busca en google por operadores aritmeticos y operadores de asignacion.

Ten cuidado, ya que nunca guardaste lo ingresado por el usuario, ya que lo pásate como parámetro directamente a mostrar solución, supongo que estabas haciendo pruebas, recuerda cambiarlo.

Valida en pedirSolucion lo ingresado por el usuario, ya que yo podría ingresarte cualquier letra, y te recomiendo guardar los valores en un char[], te sera mas fácil trabajar con ellos.

me puedes explicar en que usaras el metodo "estaRepetido"? Respecto a este método te recomendaría cambiar el ciclo while por algo así:
Código (java) [Seleccionar]

for(i = 0; i < clave.length; i++){
       if(color == clave[i]) {
                repetido = true;
                break;
        }
}


De esta manera es mas fácil de entender. ^^

En generar clave solo tienes que crear un char[4] de las letras M, R, N, A, Z, V no?

Código (java) [Seleccionar]
public static char[] generarClave(){
           char[] posiblesClaves = new char[]{'M','R','N','A','Z','V'};
           Random rnd = new Random();
           char [] clave = new char[4];
           int numeroRandom;
           for(int i = 0;i<clave.length;i++){
               numeroRandom = (int)rnd.nextDouble()*posiblesClaves.length;
               clave[i] = posiblesClaves[numeroRandom];
           }
           return clave;
       }


Así sera mas facil de entender creo yo. (verifica lo del numeroRandom que no recuerdo muy buen lo de generar números random, pero creo que entrega un numero de entre 0 5 (que es la cantidad de letras posibles))

Bien, ahora si para verificar cuantos coinciden no te entregare el código completo solo te explicare una posible manera.

Crea dos ciclos for anidados, el exterior sera el indice de colores generados por el programa y el interior sera el de colores ingresados por el usuario

Código (java) [Seleccionar]
for(int a = 0;a<generados.lengt;a++){
   for(int b = 0; b<ingresdos.lengt; b++){
          //Con esos dos indices acá podrás efectuar una comparación de cada elemento.
   }
}


Crea dos contadores, utiliza uno para cuando esta la posición y color correcto y otro para cuando el color esta correcto. Espero que te sea de ayuda, y suerte. (cualquier duda dime)

Usuario Invitado

#2
Antes que nada, el método pedirSolución debería ser así:

Código (java) [Seleccionar]
public static char[] pedirSolucion()  {
char[] solucion = new char[4];
Scanner input = new Scanner(System.in);
System.out.println("Escriba la soluccion que crea que es correcta: ");
for (int i = 0; i < solucion.length; i++) {
solucion[i] = input.next().charAt(0);
}
return solucion;
}


Pedimos que se ingrese una línea de texto y se aplica el método charAt(0) que extrae y devuelve el primer carácter del texto ingresado. Así trabajas todo con arreglos char en lugar de que la solución sea un array String.

Bien, una vez explicado ésto, creamos un método que obtenga los aciertos completos (posición y color):

Código (java) [Seleccionar]
public Map<Integer, Character> getFullMatches(char[] key, char[] solution) {
// aciertos completos: posicion y color
Map<Integer, Character> fullMatches = new HashMap<>();
for(byte i=0; i<key.length; i++) {
if(key[i] == solution[i])
fullMatches.put(i,solution[i]);
}
       return fullMatches;
}


Map<Key, Value> es un tipo de dato que guarda valores asociados, es decir, a posición de la ficha vendría a ser la llave, y su valor el color:

Posicion: Color

Un Map, List o en su defecto Collection, trabaja con Objetos. Debes saber que cada tipo de dato primitivo (char, int, byte, double, float) tiene su Wrapper, que es una clase que engloba éstos datos, permitiéndote extender su funcionalidad al ser objetos. La JVM hace lo que se llama upcasting que es convertir un primitivo a su Wrapper. Es por ésto, que aunque el objeto Map recibe un Character, le podemos pasar un char, porque la JVM lo convertirá a Character automáticamente. Sucede lo inverso con el downcasting.

Ahora, un método para los aciertos de color, que vendrían a ser fichas descolocadas. Como no nos interesa la posición del color porque es obvio, solo guardamos el color de la ficha. Por eso en lugar de un Map, utilizamos un List, que como su nombre lo indica viene a ser una lista.

Código (java) [Seleccionar]
public List<Character> getColorMatches(char[] key, char[] solution) {
// aciertos solo de color
List<Character> colorMatches = new ArrayList<>();
for(byte i=0; i<key.length; i++) {
for(byte k=0; k<solution.length; k++) {
if(key[i] == solution[k])
colorMatches.add(solution[k]);
}
}
return colorMatches;
}


Ahora sólo comprobamos si hay aciertos completos y de colores solamente. Para ésto, primero verificamos que el objeto Map devuelto por los métodos anteriores no esté vacío. Si lo está quiere decir que no se han encontrado aciertos.
Código (java) [Seleccionar]

// comprobamos si hay aciertos completos
if(!getFullMatches().isEmpty()) {
System.out.println("Se han encontrado aciertos completos: \n");
for (Map.Entry<String, String> entry : getFullMatches().entrySet()) {
System.out.println("Posición: "+entry.getKey() + " | Color: " + entry.getValue());
}


}

// comprobamos fichas descolocadas (aciertos de color)
if(!getColorMatches().isEmpty()) {
System.out.println("Se han encontrado fichas descolocadas: \n");
for (Character c : getColorMatches()) {
System.out.println("Color de la ficha: " + c);
}
}


Quizás te resulte un poco complejo pero no lo es. Si tienes dudas de algún método, nada mejor como ver la documentación del mismo en la API de Java (la encuentras en Google  ;)). Salu2!
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein