Foro de elhacker.net

Programación => Programación C/C++ => Mensaje iniciado por: niraikanai en 2 Marzo 2015, 20:27 pm



Título: Leer archivo de texto
Publicado por: niraikanai en 2 Marzo 2015, 20:27 pm
Buen día a la comunidad!

Estuve intentando leer un archivo de texto con el siguiente formato:

Pablo#10#7/12/1990

Estoy empezando a llegar a la conclusión de que no es posible leerlo con un solo fscanf con los siguientes parámetros: fscanf(pF,"%s#%d/%d/%d",nombre,&dia,&mes,&ano);

¿Me podrían dar alguna guía sobre como encarar este problema?

Saludos!

Y muchísimas gracias por su tiempo.


Título: Re: Leer archivo de texto
Publicado por: ivancea96 en 2 Marzo 2015, 20:29 pm
Lee la linea entera, y luego trabajas con ella.

La recorres hasta encontrar el primer #. Coges la primera parte. La recorres hasta el próximo #. Coges el número. Etc, etc.


Título: Re: Leer archivo de texto
Publicado por: niraikanai en 2 Marzo 2015, 20:34 pm
Gracias por la pronta respuesta.

Te referís a por ejemplo:
leerla con un letra= fgetc(pF) caracter por caracter hasta el # y cuando llego a la parte de la fecha casteo el numero para pasarlo a un int?


Título: Re: Leer archivo de texto
Publicado por: ivancea96 en 2 Marzo 2015, 20:35 pm
Por ejemplo. Aunque mejor si lees primero la linea entera. Así ahorras tiempo.


Título: Re: Leer archivo de texto
Publicado por: niraikanai en 2 Marzo 2015, 21:19 pm
Leo la linea con fgets. Por ejemplo fgets(linea, 80, pF)

Pero lo que no se me ocurre es como empezar a separar los distintos "campos". Alguna pista?


Título: Re: Leer archivo de texto
Publicado por: ivancea96 en 3 Marzo 2015, 10:21 am
Código
  1. int i = 0;
  2. char* line;
  3. /* Read from file */
  4. while(line[i]!='\0' && line[i]!='#')
  5.    ++i;

Al final de ese while, 'i' será el número de letras a coger para la primera cadena.

Código
  1. int j = 0;
  2. char name[i+1];
  3. while(j<i){
  4.    name[j] = line[j];
  5.    ++j;
  6. }
  7. name[i] = '\0';

Así, basta copiar la cadena al nuevo contenedor.


Título: Re: Leer archivo de texto
Publicado por: niraikanai en 3 Marzo 2015, 10:44 am
Había resuelto antes de ir a dormir el ejercicio de otra forma. Pero es bastante más largo que tu solución. Consistía en hacer un ciclo while con fgetc. Pero debía utilizar 3 ciclos para captar toda una línea. En cambio, con tu método, es mucho más corto.

Me ha quedado clarísimo.

Muchas gracias!


Título: Re: Leer archivo de texto
Publicado por: rir3760 en 5 Marzo 2015, 16:06 pm
Estuve intentando leer un archivo de texto con el siguiente formato:

Pablo#10#7/12/1990

Estoy empezando a llegar a la conclusión de que no es posible leerlo con un solo fscanf con los siguientes parámetros:
Código:
fscanf(pF,"%s#%d/%d/%d",nombre,&dia,&mes,&ano);
Si tienes garantizada la estructura del archivo de texto puedes leer los cinco campos de cada registro con fscanf, un ejemplo de ello pero utilizando sscanf y un array para explicarlo mejor y de una manera mas fácil (eso espero) es:
Código
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3.  
  4. int main(void)
  5. {
  6.   char *linea[] = {
  7.      "Hugo#14#7/12/1990",
  8.      "Paco#12#5/10/1988",
  9.      "Luis#10#3/08/1986",
  10.      "LINEA_SIN_EL_FORMATO_VALIDO"
  11.   };
  12.  
  13.   char nombre[100];
  14.   int num;
  15.   int d;
  16.   int m;
  17.   int a;
  18.   int i;
  19.  
  20.   for (
  21.      i = 0;
  22.      sscanf(linea[i], " %[^#]#%d#%d/%d/%d", nombre, &num, &d, &m, &a) == 5;
  23.      i++
  24.   )
  25.      printf("%s %d %d/%d/%d\n", nombre, num, d, m, a);
  26.  
  27.   return EXIT_SUCCESS;
  28. }

Otra opción es:
1) Leer cada linea con fgets.
2) Obtener cada uno de los componentes (separados por el carácter '#') con strtok.
3) Obtener los enteros del segundo y tercer componente con sscanf.

Un saludo


Título: Re: Leer archivo de texto
Publicado por: boy-ka en 1 Abril 2015, 21:54 pm
Y si en caso de que el segundo dato no sea un entero, y sea un string? u otro tipo de dato?

 printf("%s %d %d/%d/%d\n", nombre, num, d, m, a);

Te marcaría error ya que %d hace referencia a que es un entero, por qué no mejor leer el texto entero como si fuese un string?, de esta manera

Código
  1. #include <stdio.h>
  2. #include <iostream>
  3.  
  4. using namespace std;
  5.  
  6. int main(){
  7.       char datos_leidos[100];
  8.       FILE * archivo;
  9.       archivo = fopen("archivoparaleer.txt","r");
  10.       if(archivo != NULL){
  11.           fgets(datos_leidos, 100, archivo);
  12.           cout <<datos_leidos <<endl;
  13.       }else{
  14.           cout <<"No se ha podido leer el archivo" <<endl;
  15.       }
  16.       getchar();          
  17.       return 0;
  18. }
  19.  


Título: Re: Leer archivo de texto
Publicado por: rir3760 en 2 Abril 2015, 03:30 am
Y si en caso de que el segundo dato no sea un entero, y sea un string? u otro tipo de dato?

 printf("%s %d %d/%d/%d\n", nombre, num, d, m, a);

Te marcaría error ya que %d hace referencia a que es un entero
Por partes (todo lo que sigue es en buen plan):

1) El valor de retorno de las funciones scanf/fscanf/sscanf es el numero de conversiones realizadas con éxito.
2) En mi programa de ejemplo se verifica que se puedan leer las cinco conversiones, para ello es tan sencillo como verificar que el valor de retorno de la función sea igual a ... cinco.
3) Cuando se encuentra una linea que no cumple con ese formato (y eso sucede con el ultimo elemento del array, la cadena "LINEA_SIN_EL_FORMATO_VALIDO") la función retorna algún otro valor, la condición:
Código
  1. sscanf(linea[i], " %[^#]#%d#%d/%d/%d", nombre, &num, &d, &m, &a) == 5
Resulta en el valor cero (falso en C) y con ello el bucle termina su ejecución.

----

por qué no mejor leer el texto entero como si fuese un string?, de esta manera
En su tercer mensaje el creador del tema (me refiero al usuario niraikanai ) indica que utiliza esa función para leer las lineas de texto:
Leo la linea con fgets. Por ejemplo fgets(linea, 80, pF)

Un saludo


Título: Re: Leer archivo de texto
Publicado por: boy-ka en 2 Abril 2015, 07:42 am
Por partes (todo lo que sigue es en buen plan):

1) El valor de retorno de las funciones scanf/fscanf/sscanf es el numero de conversiones realizadas con éxito.
2) En mi programa de ejemplo se verifica que se puedan leer las cinco conversiones, para ello es tan sencillo como verificar que el valor de retorno de la función sea igual a ... cinco.
3) Cuando se encuentra una linea que no cumple con ese formato (y eso sucede con el ultimo elemento del array, la cadena "LINEA_SIN_EL_FORMATO_VALIDO") la función retorna algún otro valor, la condición:
Código
  1. sscanf(linea[i], " %[^#]#%d#%d/%d/%d", nombre, &num, &d, &m, &a) == 5

Resulta en el valor cero (falso en C) y con ello el bucle termina su ejecución.

----
En su tercer mensaje el creador del tema (me refiero al usuario niraikanai ) indica que utiliza esa función para leer las lineas de texto:
Un saludo

Ahh vale, es que el archivo que va a leer el programa está predefinido para siempre tener esos valores, no sabía xd.