Error al guardar muchos datos en arrays multidimensional (lenguaje C)

Iniciado por alejandrodiaz, 30 Julio 2015, 10:14 AM

0 Miembros y 3 Visitantes están viendo este tema.

alejandrodiaz

Buenas, estoy aprendiendo a programar en C y tengo muchas dudas con respecto a guardar datos en los arreglos multidimensionales.

// Tengo un ejercicio que dice: Un instituto desea controlar los resultados de los alumnos en las diferentes asignaturas de sus cursos de informatica, el programa debe leer las calificaciones obtenidas en las diferentes asignaturas y visualizar en pantalla el numero de cada estudiante seguipo por la media del estudiantes y una lista de las calificaiones parciales de ese estudiante.

el programa visualizará tambien la calificacion media de todos los estudiantes en cada asignatura. //


Es la primera vez que hago un ejercicio como este, siempre he hecho cosas sencillas, el caso es que mi codigo va bien hasta el punto en que tengo que guardar las notas en los arrays, ya que en el modo en que lo hago se me sobreescribe las notas de un alumno sobre otra.  Para el instituto que lo hago no es por semestres sino por trimestres, tres trimestres en un año.

Al final me da datos erroneos, de otras partes de la memoria. Se que deberia usar punteros y funciones pero como dije soy novato y me confundo. Quiero ver primero que funciona y luego darle orden.

como seria la forma correcta de guardar las notas para cada alumno y materia?

este es mi codigo



#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <conio.h>

//Declarando variables globales para que el usuario establezca limites en los ciclos FOR//
int cant_alum;
int cant_mate;
int cant_nota;

int main()
{
   //Arrays para nombres de alumnos y materias ---Arrays de caracteres---//
   char nom_alum[cant_alum][30];
   char nom_mate[cant_mate][40];

   //Arrays para notas de trayectos ---not_tra1 = notas del trayecto1--- //
   
   int  not_tra1[4][cant_nota];    
   int  not_tra2[4][cant_nota];
   int  not_tra3[4][cant_nota];

   //Llamando a las variables globales//
   cant_alum;
   cant_mate;
   cant_nota;

   //Enteros usados en los ciclos FOR//
   int i;
   int j;
   int k;


//Inicia el programa//
       printf("Cuantos alumnos desea registrar?: ");
       fflush(stdin);
       scanf("%d", &cant_alum);

           for (i=1; i<cant_alum+1; i++){
               printf("Alumno nro %d: ",i);
               fflush(stdin);
               gets(nom_alum[i]);
   }

       printf("\nCuantas materias desea registrar?: ");
       fflush(stdin);
       scanf("%d",&cant_mate);

           for(i=1; i<cant_alum+1; i++){
               printf("Materia nro %d: ",i);
               fflush(stdin);
               gets(nom_mate[i]);
           }

   printf("\nCuantas evaluaciones fueron por trayecto?: ");
   fflush(stdin);
   scanf("%d", &cant_nota);


//PARTE 2: GUARDAR DATOS. MAL HECHO//

           for(i=1; i<cant_alum+1; i++){
               printf("\n\n********** Alumno nro %d : %s **********\n" , i, nom_alum[i]);

                       for(j=1; j<cant_mate+1; j++){
                           printf("\n\nIngrese las notas parciales de %s\n", nom_mate[j]);

                               printf("\n===== Trayecto 1 =====\n\n");
                                   for (k=1; k<cant_nota+1; k++){
                                       printf("Nota %d: ", k);
                                       fflush(stdin);
                                       scanf("%d",&not_tra1[k]);
                           }
                               printf("\n===== Trayecto 2 =====\n\n");
                                   for (k=1; k<cant_nota+1; k++){
                                       printf("Nota %d: ", k);
                                       fflush(stdin);
                                       scanf("%d",&not_tra2[k]);
                           }
                               printf("\n===== Trayecto 3 =====\n\n");
                                   for (k=1; k<cant_nota+1; k++){
                                       printf("Nota %d: ", k);
                                       fflush(stdin);
                                       scanf("%d",&not_tra3[k]);
                           }
                       }
                   }

printf("Esto se muestra sin error");

//Aca si aplico ciclos FOR con printf para mostrar los valores guardados en los arreglos, todos los datos mostrados son erroneos//


getch();
return 0;
}



Al analizar pienso que guardar las notas de la forma ejemplo:

scanf("%d", &not_tra1 [k]);  no está bien hecho, creo que deberia ser &not_tra1
  •  ... donde "X" sea la variable pero no se como organizarlo, de las maneras que las he hecho todas malas.

    tambien para los arreglos de trayectos tengo

    not_tra1[4][cant_nota]; Donde 4 quise definir la cantidad de trayectos pero se ahora se que es un error porque no dejo crecer la matriz en filas, quisiera que sea automatica pero dejando [ ] vacio me tira error.


    Y por ultimo veo que guardar todas las notas de los mismos alumnos y materias de un trayecto en un solo arreglo  es una cantidad de numeros desorganizada.. pero con tanto lio no se como avanzar

    Espero me puedan ayudar, soy muy novato como ven... Gracias!

joecarl

#1
Esta parte del código está mal:

Código (cpp) [Seleccionar]

//Arrays para nombres de alumnos y materias ---Arrays de caracteres---//
   char nom_alum[cant_alum][30];
   char nom_mate[cant_mate][40];
//Arrays para notas de trayectos ---not_tra1 = notas del trayecto1--- //

   int  not_tra1[4][cant_nota];    
   int  not_tra2[4][cant_nota];
   int  not_tra3[4][cant_nota];

   //Llamando a las variables globales//
   cant_alum;
   cant_mate;
   cant_nota;


No puedes definir el tamaño de un array con una variable, es decir:
Código (cpp) [Seleccionar]
char nom_alum[cant_alum][30];
char nom_mate[cant_mate][40];
int not_tra1[4][cant_nota];


Deberia ser:
Código (cpp) [Seleccionar]
char nom_alum[10][30];
char nom_mate[10][40];
int not_tra1[4][10];


He puesto 10, pero eso no quiere decir que haya q usar las 10, sino que como maximo tenemos 10.

Otra cosa es que si ya estas creando varios arrays para cada trayecto entonces el [4] sobra, aunque quedaria más organizado en un solo array, es decir, en lugar de:
Código (cpp) [Seleccionar]

   int  not_tra1[4][cant_nota];    
   int  not_tra2[4][cant_nota];
   int  not_tra3[4][cant_nota];


Deberías poner:
Código (cpp) [Seleccionar]

   int  not_tra[4][10];    


Luego esta parte:
Código (cpp) [Seleccionar]

//Llamando a las variables globales//
   cant_alum;
   cant_mate;
   cant_nota;

No sirve para nada, no tiene efecto, esas lineas las puedes eliminar directamente.

Cuando entiendas eso respónde al post y seguimos.

alejandrodiaz

#2
Perfecto, ya he modificado el arreglo de trayecto, y los char lo he puesto. voy a seguir con los valores de tu ejemplo y luego los modifico a los que requiera.


char nom_alum[10][30];
char nom_mate[10][40];
int  not_tra[4][10];


habia puesto 4 porque en los FOR iniciaba con el i=1, para en el printf usar el "i" para ordenar los mensajes por el numero de alumnos ejemplo: alumno nro 1; igual veo que es un desperdicio de elemento.

También quite las variables globales del main.. estaba apenas aprendiendo la diferencia entre variables locales y globales. Gracias por la aclaratoria.

joecarl

#3
Perfecto, seguimos:

Código (cpp) [Seleccionar]

fflush(stdin);

Esto tampoco tiene efecto, fflush sirve para vaciar los buffer de salida, los de entrada no puedes vaciarlos de esta manera. Asique elimina también estas líneas.

Si solo quieres 3 trayectos pues declara solo 3:
Código (cpp) [Seleccionar]

int  not_tra[3][10];


Y el primer trayecto sera el 0 el segundo el 1 y el tercero el 2. Acostumbrate a reccorrer los arrays de esa manera, así es como se debe hacer siempre en C.

Con respecto a esto:

Código (cpp) [Seleccionar]

                       for(j=1; j<cant_mate+1; j++){
                           printf("\n\nIngrese las notas parciales de %s\n", nom_mate[j]);

                               printf("\n===== Trayecto 1 =====\n\n");
                                   for (k=1; k<cant_nota+1; k++){
                                       printf("Nota %d: ", k);
                                       fflush(stdin);
                                       scanf("%d",&not_tra1[k]);
                           }
                               printf("\n===== Trayecto 2 =====\n\n");
                                   for (k=1; k<cant_nota+1; k++){
                                       printf("Nota %d: ", k);
                                       fflush(stdin);
                                       scanf("%d",&not_tra2[k]);
                           }
                               printf("\n===== Trayecto 3 =====\n\n");
                                   for (k=1; k<cant_nota+1; k++){
                                       printf("Nota %d: ", k);
                                       fflush(stdin);
                                       scanf("%d",&not_tra3[k]);
                           }
                       }
                   }


Lo puedes sustituir por:

Código (cpp) [Seleccionar]

   int tra;//variable contadora para el bucle de trayectos, la puedes declarar en otro sitio si quieres.
   for(j=1; j<cant_mate+1; j++){
       printf("\n\nIngrese las notas parciales de %s\n", nom_mate[j]);
       for(tra=0; tra<3; tra++){
            printf("\n===== Trayecto %d =====\n\n",tra+1);
            for (k=0; k<cant_nota; k++){
                 printf("Nota %d: ", k+1);            
                 scanf("%d",&not_tra[tra][k]);
            }              
       }
  }


Fijate en como cambia la forma de recorrer el array:
Código (cpp) [Seleccionar]

for (k=1; k<cant_nota+1; k++) //mal
for (k=0; k<cant_nota; k++) //bien





alejandrodiaz

Buenisimo, ya voy entendiendo amigo.

Lo de fflush(stdin) lo habia puesto porque al principio tengo un scanf y luego le sigue un gets, entonces al compilar me saltaba un ciclo en for, lei algo sobre que el scanf al finalizar agrega un salto de line \n y el gets lo leia y guardaba el espacio en un elemento del arreglo.


       printf("Cuantos alumnos desea registrar?: ");
       fflush(stdin);
       scanf("%d", &cant_alum);      // <=== aca tengo el scanf//

           for (i=1; i<cant_alum+1; i++){
               printf("Alumno nro %d: ",i);  //<===y aqui al compilar en vez de "Alumno nro 1, me saltaba a Alumno nro 2"//
               fflush(stdin);
               gets(nom_alum[i]);    //<=== este el gets//



Ya empiezo a corregir el resto.

Con esto podré ya imprimir todos los datos agregados o todavia falta?

joecarl

Sí lo del retorno del línea es cierto, pero es que fflush(stdin); no deberia solucionarlo. ¿A ti te funciona? Porque eso es muy raro que funcione, pero bueno si te va bien así déjalo, aunque ya te digo que a mi eso nunca me ha funcionado, la solucion correcta es otra más compleja, si quieres más adelante te la explico.

Y sí, yo creo que con eso ya estaría todo arreglado, si no te funciona y no encuentras el error pues coméntalo.

alejandrodiaz

Si bueno, en ese caso que mencione me funciono y no molesto mas, por eso lo puse en todos lados para prevenir.

Cuando tengas oportunidad explicas la solucion para no usar fflush(stdin) todo lo que ayude en mi aprendizaje es bienvenido.

Y claro seguro seguire comentando, me toca aplicar funciones, punteros y sacar los promedios, estoy casi seguro que tendre muchos errores.

Muchas gracias amigo por la ayuda, seguire editando el codigo.

joecarl

#7
La forma adecuada de leer enteros por teclado sin dar pie a problemas es así:

Código (cpp) [Seleccionar]

int num;
char buffer[100];//cadena de texto auxiliar para copiar el buffer de entrada
//------
fgets(buffer,100,stdin); //copiamos el buffer de entrada a la cadena auxiliar
sscanf(buffer, "%d", &num); //leemos el entero de la cadena auxiliar

Esto se hace precisamente para quitar ese retorno de linea que nos está haciendo la puñeta, ya que con fgets estamos vaciando stdin por completo.

EDIT: también puedes meter ese código en una función para no tener que andar repitiéndolo todo el rato:

Código (cpp) [Seleccionar]

int leerEntero(){
   int num;
   char buffer[100];//cadena de texto auxiliar para copiar el buffer de entrada
   fgets(buffer,100,stdin); //copiamos el buffer de entrada a la cadena auxiliar
   sscanf(buffer, "%d", &num); //leemos el entero de la cadena auxiliar
   return num;
}


También añado que leer una cadena de texto con gets es inseguro ya que gets no controla el numero máximo de caracteres que vas a leer, por tanto deberías usar fgets también. Es decir, en lugar de:

Código (cpp) [Seleccionar]

gets(nom_alum[i]);


Usa:

Código (cpp) [Seleccionar]

fgets(nom_alum[i],30,stdin);


Por ultimo, decirte que he puesto 30 pero lo correcto es usar un DEFINE arriba del todo para no tener que andar poniendo 30 por todas partes, es decir:
Código (cpp) [Seleccionar]

#define MAX_CHARS 30 //esto arriba del todo
int main(...){
//.....
char nom_alum[10][MAX_CHARS];
//.....otras declaraciones....
//.....hacer tareas.....

fgets(nom_alum[i], MAX_CHARS,stdin);
//.....
}