Leer archivo de texto

Iniciado por niraikanai, 2 Marzo 2015, 20:27 PM

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

niraikanai

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.

ivancea96

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.

niraikanai

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?

ivancea96

Por ejemplo. Aunque mejor si lees primero la linea entera. Así ahorras tiempo.

niraikanai

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?

ivancea96

int i = 0;
char* line;
/* Read from file */
while(line[i]!='\0' && line[i]!='#')
    ++i;


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

int j = 0;
char name[i+1];
while(j<i){
    name[j] = line[j];
    ++j;
}
name[i] = '\0';


Así, basta copiar la cadena al nuevo contenedor.

niraikanai

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!

rir3760

Cita de: niraikanai en  2 Marzo 2015, 20:27 PMEstuve 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);
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:
#include <stdio.h>
#include <stdlib.h>

int main(void)
{
   char *linea[] = {
      "Hugo#14#7/12/1990",
      "Paco#12#5/10/1988",
      "Luis#10#3/08/1986",
      "LINEA_SIN_EL_FORMATO_VALIDO"
   };
   
   char nombre[100];
   int num;
   int d;
   int m;
   int a;
   int i;
   
   for (
      i = 0;
      sscanf(linea[i], " %[^#]#%d#%d/%d/%d", nombre, &num, &d, &m, &a) == 5;
      i++
   )
      printf("%s %d %d/%d/%d\n", nombre, num, d, m, a);
   
   return EXIT_SUCCESS;
}


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
C retains the basic philosophy that programmers know what they are doing; it only requires that they state their intentions explicitly.
--
Kernighan & Ritchie, The C programming language

boy-ka

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 (cpp) [Seleccionar]

#include <stdio.h>
#include <iostream>

using namespace std;

int main(){
       char datos_leidos[100];
       FILE * archivo;
       archivo = fopen("archivoparaleer.txt","r");
       if(archivo != NULL){
           fgets(datos_leidos, 100, archivo);
           cout <<datos_leidos <<endl;
       }else{
           cout <<"No se ha podido leer el archivo" <<endl;
       }
       getchar();         
       return 0;
}

rir3760

Cita de: boy-ka en  1 Abril 2015, 21:54 PMY 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:
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.

----

Cita de: boy-ka en  1 Abril 2015, 21:54 PMpor 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:
Cita de: niraikanai en  2 Marzo 2015, 21:19 PMLeo la linea con fgets. Por ejemplo fgets(linea, 80, pF)

Un saludo
C retains the basic philosophy that programmers know what they are doing; it only requires that they state their intentions explicitly.
--
Kernighan & Ritchie, The C programming language