Foro de elhacker.net

Programación => Programación C/C++ => Mensaje iniciado por: Sayuri108 en 6 Enero 2018, 15:00 pm



Título: Problemas con los void
Publicado por: Sayuri108 en 6 Enero 2018, 15:00 pm
Hola!
Estoy haciendo un programa en el que uso void y arrays, y no se por que en cuanto llego a un void el array cambia de tamaño  :huh:
Este es el punto deonde me da problemas:

Código
  1. const int tamaño = 4;
  2.  
  3. typedef int tCode[tamaño];
  4.  
  5. void codigoAleatorio(tCode code) {    //Aqui cambia a ser code ={0} en vez de code = {0,0,0,0}
  6. int random = 0;
  7. srand(time(NULL));
  8. for (int i = 0; i < TAM_CODIGO; i++) {
  9. random = rand() % 7;
  10. code[i] = random; (aqui no se por que no salen [] con la i dentro despues de code)
  11. }
  12. }
  13.  
  14. el main esta asi:
  15. int main(){
  16.     tCode code = {};
  17.      /*funcion*/
  18. }

alguna idea?
Gracias.


Título: Re: Problemas con los void
Publicado por: engel lex en 6 Enero 2018, 15:08 pm
Citar
(aqui no se por que no salen [] con la i dentro despues de code)

porque debes usar las etiquetas GeSHi, si no [ i ] significa cursiva, porfa corrige tu codigo para ver bien el problema y en el main si es posble pon más codigo relacionado


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 6 Enero 2018, 15:18 pm
vale, gracias.
Este es mi main:
Código
  1. int main() {
  2. int opcion = menu();
  3. EjecutarMenu(opcion);
  4. return 0;
  5. }
  6. int EjecutarMenu(int opcion) {
  7. tCodigo codigo = {};
  8. tCodigo hipotesis = {};
  9. bool admiteRepetidos = false;
  10. int colocados = 0, descolocados = 0;
  11. while (opcion != 0) {
  12. switch (opcion) {
  13. case (1): {//funcion no repetidos
  14. opcion1(codigo, hipotesis, admiteRepetidos, colocados, descolocados);
  15. }
  16.  break;
  17. case (2): {//funcion con repetidos
  18. opcion2(codigo, hipotesis, admiteRepetidos, colocados, descolocados);
  19. }
  20.  break;
  21. }
  22. }
  23. return opcion;
  24. }

las funciones opcion1 y opcion2 son dos void bastante parecidos y en cuanto llega al void cambia el  tamaño del array.

Código
  1. void opcion1(tCodigo codigo, tCodigo hipotesis, bool admiteRepetidos, int colocados, int descolocados) {
  2. admiteRepetidos = false;
  3. codigoAleatorio(codigo, admiteRepetidos);
  4. cout << "Introduce el codigo (palabra de 4 letras con alguna de las siguientes, R, Z, V, A, M, B): ";
  5. for (int s = 0; s < TAM_CODIGO - 1; s++)
  6. cin >> hipotesis[s];
  7. compararCodigos(codigo, hipotesis, colocados, descolocados);
  8. ImprimirPantalla(codigo, hipotesis, colocados, descolocados);
  9. while (colocados != TAM_CODIGO)
  10. cout << "Introduce el codigo (palabra de 4 letras con alguna de las siguientes, R, Z, V, A, M, B): ";
  11. for (int s = 0; s < TAM_CODIGO - 1; s++)
  12. cin >> hipotesis[s];
  13. compararCodigos(codigo, hipotesis, colocados, descolocados);
  14. ImprimirPantalla(codigo, hipotesis, colocados, descolocados);
  15. }


Título: Re: Problemas con los void
Publicado por: MAFUS en 6 Enero 2018, 16:30 pm
Tienes algunos fallos que atribuyó a que aunque no conoces del todo el lenguaje y buenas prácticas, pero me gustaría que explicarás en qué notas que el array cambia el tamaño.
Lo que si he observado es que haces cosas así:

Código:
for (int s = 0; s < TAM_CODIGO - 1; s++)

TAM_CODIGO marca el número de elementos, como C empieza por el 0 el máximo índice será TAM_CODIGO -1, y precisamente ese es al que va a llegar el for con la sentencia s<TAM_CODIGO: mientras s esté por debajo de TAM_CODIGO se va a ejecutar su código asociado. Pero tú le restas 1 a TAM_CODIGO con lo que no ejecutará las veces suficientes el código asociado, te faltará una. Tal vez eso es lo que te pasa.


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 6 Enero 2018, 16:47 pm
Código:
for (int s = 0; s < TAM_CODIGO - 1; s++)

es que comparo dos cosas en las posiciones del array, entonces no quiero que se salga del array (comparar el ultimo elemento del array con la nada), el ultimo elemento de array solo se compara con su anterior, entonces x solo puede llegar a valer como maximo 3. Pero el problema es que en cuanto pasa a un void, el array cambia automaticamente de tamaño, en vez de seguir siendo de tamaño cuatro, pasa a ser de tamaño 1.

ahi dejo una captura de como se cambia el solito
https://gyazo.com/3643d198efb64fb42a8b85c5cb4c50bf (https://gyazo.com/3643d198efb64fb42a8b85c5cb4c50bf)
https://gyazo.com/072244555cacd0ef0a6fb45f4a39485b (https://gyazo.com/072244555cacd0ef0a6fb45f4a39485b)


Título: Re: Problemas con los void
Publicado por: MAFUS en 6 Enero 2018, 18:24 pm
Vale. El void no es lo que pasas sino el tipo de dato devuelto de la función. Por otra parte te diría que mires un paper de lo que ocurre cuando se usa un array como argumento.

A grandes rasgos un array de una dimensión es lo mismo que un puntero, solo se pasará la dirección del primer elemento, todos los demás 'adornos' como el tipo de dato del array y loas restantes dimensiones que pueda tener (se especifican todas las dimensiones de un array menos la de más a la izquierda) solo sirven para, internamente, hacer aritmética y encontrar el dato deseado cuando usas notación de array.

Por tanto un array, cuando es pasado a una función, pierde toda la información de su tamaño y, a ojos del compilador, es un puntero.


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 6 Enero 2018, 18:27 pm
entonces como puedo hacerlo?
por que me piden que lo haga asi (es un trabajo de clase) y por narices tengo que usar voids,de hecho me ponen algunos asi:
Código
  1. void codigoAleatorio(tCodigo codigo, bool admiteRepetidos);
y tengo que rellenarlos, otros solo me dan el nombre de la funcion.
y los que me dan "completos" no puedo modificarlos
asi que por ejemplo no puedo ponerle :
Código
  1. void codigoAleatorio(const tCodigo codigo, bool admiteReetidos);


Título: Re: Problemas con los void
Publicado por: MAFUS en 6 Enero 2018, 18:53 pm
Olvídate del void. En este caso indica que la función no devuelve ningún valor, solo hace un trabajo (en otros lenguajes se llamaría procedimiento o subprograma).

Cuando se pasa un array a una función lo normal es pasar también el número de elementos que el array contiene en otro argumento y así la función conocerá el límite. De todas formas tú usas una variación de eso qué es mediante una variable global llamada TAM_CODIGO, que por otra parte, en el primer código que has mostrado la llamas 'tamaño'.

Es decir, dentro de la función debes olvidarte del tamaño del array, lo verás como si fuera un puntero (o un entero sin signo, por si no has llegado a los punteros) y usa para trabajar con él los límites, ya sean variables globales o argumentos que lo van a acompañar en las llamadas a funciones.


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 6 Enero 2018, 18:58 pm
entonces a la vuelta de vacaciones tendre que hablar con el profesor, por que quiere que use esa funcion y que haga algo determinado y no me puedo salir del guion...
Gracias por la ayuda


Título: Re: Problemas con los void
Publicado por: MAFUS en 6 Enero 2018, 19:22 pm
¿Exactamente qué te pide exactamente?


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 6 Enero 2018, 20:11 pm
void codigoAleatorio(tCodigo codigo, bool admiteRepetidos): elige
aleatoriamente un código y lo devuelve con el parámetro de salida. El
segundo parámetro permite indicar si se admiten colores repetidos en él.

para eso nos dio tambien como hacer  una funcion que genere numeros aleatorios y con esa funcion rellenamos el array dentro del void.


Título: Re: Problemas con los void
Publicado por: MAFUS en 6 Enero 2018, 23:28 pm
Bien.
Mira:
Lo primero de todo, para entender lo que debes hacer es entender qué te pide la función, así que voy a desglosarla:
void: Como te he dicho indica que la función no retorna valor alguno, solo realiza un trabajo.
codigoAleatorio: Tú dices que debe generar un código aleatorio en el array que se le va a pasar. De ahí que tenga ese nombre.
tCodigo codigo: Es el array que le pasarás a esta función. Ahora algo que seguramente no has visto: Los arrays y punteros se dice que se pasan por referencia, es decir, ese dato se podrá cambiar. El funcionamiento interno es un poco más complicado, pero por ahora te basta saber eso. Así es como la función retornará el array relleno.
bool admiteRepetidos: Dato que indicará si se pueden repetir los elementos.

Como puedes ver no hay ningún argumento que diga el tamaño que tendrá el array, así que lo deberás tomar de la constante que usaste para declararlo: TAM_CODIGO.

Deduzco que la función que te ha dado el profesor es la de la librería estándar rand().

Deduzco, partiendo de lo que has ido mostrando, es que el código se realiza a partir de una combinación de las letras R, Z, V, A, M, B.

Así que empiezo a escribir:
Código
  1. void codigoAleatorio(tCodigo codigo, bool admiteRepetidos) {
  2.    unsigned i; // mi índice para el array.
  3.    unsigned j; // mi índice para un array auxiliar, ver mas abajo.
  4.    char c; // tendrá el elemento del código seleccionado.
  5.    char elementosCodigo[] = {'R', 'Z', 'V', 'A', 'M', 'B'}; // el array que contiene los elementos que conforman el código.
  6.    const unsigned TAM_ELEM_COD = 6; // 6 es el tamaño del array elementosCodigo. En verdad se debería sustituir 6 por 'sizeof(elementosCodigo) / sizeof(char)', pero por razones de que todavía no habrás avanzado lo suficiente en el curso lo dejo con el 6.
  7.  
  8.    // Ahora voy a rellenar el array codigo que ha recibido la función.
  9.    for(i = 0; i < TAM_CODIGO; ++i) {
  10.        j = rand() % TAM_ELEM_COD;
  11.        // Ahora viene la parte donde trabajamos con si se admiten elementos repetidos o no.
  12.        if(!admiteRepetidos) { // Si no se admiten repetidos...
  13.            // La idea es cambiar el dato de elementosCodigo que ya se ha usado a 0, así sabremos que ese ya se ha pillado.
  14.            while((c = elementosCodigo[j]) == 0) { // Mientras encontremos que el elemento seleccionado ya ha sido escogido con anterioridad
  15.                ++j; // Elegiremos el siguiente elemento.
  16.                if(j >= TAM_ELEM_COD); // Si el índice j queda fuera del array de código...
  17.                    j = 0; // Se regresa el índice al inicio.
  18.            } // Terminado el while c tendrá un carácter válido y j estará sobre ese elemento que no se ha repetido.
  19.            elementosCodigo[j] = 0; // Para evitar que en posteriores búsquedas, para rellenar el array codigo, se elija de nuevo esta posición, se marca con el 0.
  20.        } // Como en el paso anterior ya tenemos un elemento del código válido podemos continuar. Nótese que si se admitieran repetidos la función es la misma, lo único que todo lo que hay dentro del if se puede omitir, por eso que se ha escrito de esta forma.
  21.        codigo[i] = c; // Se copia a la posición i del array codigo el carácter obtenido de elementosCodigo.
  22.    } // Terminado el for ya tenemos el código listo por tanto...
  23. } // fin de la función.
               


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 7 Enero 2018, 12:36 pm
Vale, creo que ya lo entiendo, muchas gracias por tu ayuda.  :)


Título: Re: Problemas con los void
Publicado por: MAFUS en 7 Enero 2018, 12:54 pm
Un placer  ;-)


Título: Re: Problemas con los void
Publicado por: srWhiteSkull en 7 Enero 2018, 22:30 pm
Si dice que tiene que devolver o retornar un código puede hacerlo con un puntero genérico (void *) y otra alternativa es modificar el array que le pasas como parámetro (por referencia) para luego modificarlo.

Ejemplo :
Código
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3.  
  4. /* prototipos */
  5. void * funcionConPunteroGenerico();
  6. void subrutina(char * cadena);
  7.  
  8. void main()
  9. {  
  10.    /* Aqui reservamos espacio de memoria a 'cadena' pero NO asignamos valores*/
  11.    char * cadena=(char *) malloc(10 * sizeof(char));
  12.  
  13.    /* Devolvemos una cadena */
  14.    printf(funcionConPunteroGenerico());
  15.  
  16.    /* Asignamos valores a la 'cadena' */
  17.    subrutina(cadena);
  18.    /* Mostramos el contenido de la 'cadena' */
  19.    printf(cadena);
  20.  
  21. }
  22.  
  23. void * funcionConPunteroGenerico(){
  24.    char * cadena =(char *) malloc(10*sizeof(char));
  25.    *cadena='H';
  26.    *(cadena+1)='o';
  27.    *(cadena+2)='l';
  28.    *(cadena+3)='a';
  29.    *(cadena+4)=' ';
  30.    *(cadena+5)='M';
  31.    *(cadena+6)='u';
  32.    *(cadena+7)='n';
  33.    *(cadena+8)='d';
  34.    *(cadena+9)='o';
  35.    *(cadena+10)=0; /* final de la cadena */
  36.  
  37.    return cadena; /* Podemos devolver cualquier tipo */
  38. }
  39.  
  40. void subrutina(char * cadena){ /* por referencia */
  41.    *cadena='A';
  42.    *(cadena+1)='d';
  43.    *(cadena+2)='i';
  44.    *(cadena+3)='o';
  45.    *(cadena+4)='s';
  46.    *(cadena+5)=0; /* final de la cadena */    
  47. }


Título: Re: Problemas con los void
Publicado por: MAFUS en 7 Enero 2018, 23:58 pm
Esto está mal.
Dentro de funcionConPunteroGenerico capturas memoria de forma dinámica y siempre que se haga algo así se debe liberar manualmente.
Tú haces que el valor devuelto por la función sea usado por printf, pero al no capturarlo con una variable ese puntero se va a perder y quedarás con esa porción de memoria inutilizada al no poder liberarla.

Lo mismo con la variable cadena. La capturas pero nunca la liberas. Es cierto que los modernos S.O. liberan la memoria ocupada por un programa cuando éste termina pero ten en cuenta de que si alguna vez vas a trabajar con uno antiguo, un embebido o con un sistema diferente; o vayas a realizar un programa que su ejecución se alarga en el tiempo, este tipo de fallos pueden llegar a hacer lento e inestable todo el sistema.


Título: Re: Problemas con los void
Publicado por: srWhiteSkull en 8 Enero 2018, 00:16 am
Bueno, pienso que "Esto está mal" no es lo correcto. Déjalo en "incompleto" porque es así como indicas, faltaría la parte que libera el espacio reservado... es lo que tiene cuando trabajas con punteros.

De todas formas el código simplemente muestra las alternativas que cité en el mensaje. Gracias MAFUS, tú comentario es bastante didáctico, esperemos que no vaya a petar mi PC de 64kb  ;-) ;-)

Código
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4.  
  5. void * funcionConPunteroGenerico();
  6. void subrutina(char * cadena);
  7.  
  8. void main()
  9. {  
  10.    /* Aqui reservamos espacio de memoria a 'cadena' pero NO asignamos valores*/
  11.    char cadena[10];
  12.    char * otraCadena=funcionConPunteroGenerico();
  13.  
  14.    /* Devolvemos una cadena */
  15.    printf("%s\n",otraCadena);
  16.  
  17.    /* Asignamos valores a la 'cadena' */
  18.    subrutina(cadena);
  19.    /* Mostramos el contenido de la 'cadena' */
  20.    printf(cadena);
  21.  
  22.    /* liberamos el espacio reservado del puntero */
  23.    free(otraCadena);    
  24.  
  25.    /* el array no es necesario ya que apunta a la pila(memoria estática) del programa
  26.        , y esta es liberada al cerrar o salir del programa. */
  27. }
  28.  
  29. void * funcionConPunteroGenerico(){
  30.    char * cadena =(char *) malloc(11*sizeof(char));
  31.    strcpy(cadena,"Hola Mundo\0");
  32.  
  33.    return cadena; /* Podemos devolver cualquier tipo */
  34. }
  35.  
  36. void subrutina(char * cadena){ /* por referencia */
  37.    strcpy(cadena,"Adios\0");
  38. }


Título: Re: Problemas con los void
Publicado por: Sayuri108 en 9 Enero 2018, 22:09 pm
hola de nuevo, al fin tuve clase con mi profesor y me dijo que lo que debia hacer era poner
Código
  1. void codigoAleatorio(tCodigo & codigo, bool admiteRepetidos);
y asi hace referencia al array que yo quiero y no me genera uno nuevo cada vez.


Título: Re: Problemas con los void
Publicado por: srWhiteSkull en 10 Enero 2018, 01:47 am
Pues es igual al ejemplo que te pasé, con subrutina, solo que yo use lenguaje c y por eso puse asterisco. Paso por referencia, trabaja mas rapido que paso por valor ya que no copia el objeto o valor sino trabaja directamente con el espacio donde se alojan los datos.