guardar nodos en txt

Iniciado por d91, 17 Abril 2014, 19:21 PM

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

d91

hola a todos, estoy haciendo mi proyecto final con programacion orientado a objetos pero no he encontrado algun tutorial de estructura de datos con archivos, pero he intentado ingeniarmelas, usando una cola dinamica para los pacientes que llegan a una clinica, he trabajado con archivos pero con estructuras no con clases aunque al parecer la diferencia es que las estructuras no tienen metodos, constructores y destructores y como son dinamicas aumenta la dificultad, se muy bien que en este codigo estoy alternando c y c++ pero lo que es archivos se me es mas facil en c. Entonces mi  dificultad es como guardar una cola dinamica en txt y despues volver a cargarla, hice algunos procedimientos para guardar pero no se si esta bien aqui el codigo

Código (cpp) [Seleccionar]

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


class nodo {
   public:
    nodo(int num, char name[20], nodo *sig = NULL)
    {
    numero=num;
    strcpy(nombre,name);
    siguiente=sig;
    }//constructor
   private:
    int numero;
    char nombre[20];
    nodo *siguiente;

    friend class cola;
};

typedef nodo *pnodo; //definimos un tipo de dato personalizaodo. Tipo puntero de la clase NODO

class cola {
   public:
       cola() : primero(NULL), ultimo(NULL) { }
    ~cola();

    void encolar(int num, char name[]);
    int desencolar();
    void Mostrar();
    void Guardar(int cont);
  // private:
   pnodo primero, ultimo;
};
cola paciente;
cola::~cola()
{
   while(primero) desencolar();
}

void cola::encolar(int num, char name[])
{
   pnodo nuevo;
   nuevo = new nodo(num,name);
   if(ultimo) ultimo->siguiente = nuevo;
   ultimo = nuevo;
   if(!primero) primero = nuevo;
}

int cola::desencolar()
{
   pnodo nodo; /* variable auxiliar para manipular nodo */
   int v;      /* variable auxiliar para retorno */

   nodo = primero;
   if(!nodo) return 0; /* Si no hay nodos en la pila retornamos 0 */
   primero = nodo->siguiente;
   //v = nodo->valor;
   delete nodo;
   if(!primero) ultimo = NULL;
   return 1;
}            /*Procedimiento Mostrar Cola*/
/*--------------------------------------------------------*/
void cola::Mostrar(){                                     //
pnodo aux;                                                //
aux=primero;                                           //
   while(aux!=NULL){                                      //
      cout<<"\n\nnumero de paciente "<<aux->numero;
      cout<<"\nNombre "<<aux->nombre;                              //
      aux=aux->siguiente;                                 //
   }                                                      //
}                                                         //

/********************Guarda en txt*************************/

void cola::Guardar(int cont){
FILE *ptr=fopen("C:\\SaveStruct.txt","at+"); /*en esta parte no se si lo estoy hacien bien*/
   fwrite(&paciente,sizeof(paciente),1,ptr);    //intento guardar los nodos como una estructura normal
   fclose(ptr);
   cout<<"Operacion exitosa";
   getch();

}

void IngresoDatos(){
   int no;
   char nombre[15];
clrscr();
cout<<"Ingrese no: ";cin>>no;
   cout<<"Ingrese nombre: ";fgets(nombre,15,stdin);
   if(nombre[strlen(nombre)-1]=='\n')

   nombre[strlen(nombre)-1]='\0';
   paciente.encolar(no, nombre);
   getch();
}

void CargarTxt(){
clrscr();
   cout<<"en construccion";
   getch();
}
int main()
{
   cola Cola1;
   cola Cola2;

   int contador=0;
   int num=0;
   int no;
   char nombre[15];
   cola call;
   do{
   clrscr();
      cout<<"\n1. Ingresar datos \n";
      cout<<"2. Mostrar cola\n";
      cout<<"3. Desencolar\n" ;
      cout<<"4. Guardar en txt\n";
      cout<<"5. Leer txt\n";
      cout<<"6. Salir";
      cout<<"\n Seleccione su opcion: ";
      cin>>num;
      switch(num){
      case 1:
      cout<<"\nIngreso de datos \n";
      IngresoDatos();
         contador=contador+1;
      break;

      case 2:
      clrscr();
       cout<<"Mostrar cola\n";
paciente.Mostrar();
       //       barajar(Cola1,Cola2);
       getch();
      break;

      case 3:
      cout<<"desencolando";
      paciente.desencolar();
      getch();
      break;

      case 4:
      cout<<"Guardando";
      paciente.Guardar(contador);
      getch();
      break;

      case 5:
      cout<<"cargando";
      CargarTxt();
      }
   }while(num!=6);

}//encola y desencola correcta el problema guardar esa cola en txt y despues cargarla

eferion

Nota 1.

Una lista enlazada es, por definición, dinámica... luego la dificultad es siempre la misma. No?

Nota 2.

Estás trabajando con C++, luego no termino de entender por qué el nombre lo almacenas con un array de chars en vez de usar la clase string.

Código (cpp) [Seleccionar]

class nodo {
  public:
   nodo(int num, const std::string& name, nodo *sig = NULL)



Nota 3.

No tiene sentido que pongas includes de C... los includes de C++ no llevan la extensión .h. La STL es suficientemente amplia y completa, no necesitas los includes de C para nada.

Nota 4.

Por qué motivo cola es clase friend de nodo??? Es mucho mejor que pongas los getters y setters correspondientes en nodo... pocas veces es necesario usar friend y este no es el caso ( además puede ser una fuente de problemas ).

Código (cpp) [Seleccionar]

class nodo {
  public:
   nodo(int num, const std::string& name, nodo *sig = NULL)
     : numero( num ),
       nombre( name ),
       siguiente( sig )
   {
   }//constructor
   
   int Numero( ) const
   { return numero; }
   
   std::string Nombre( ) const
   { return nombre; }
   
   nodo* Siguiente( ) const
   { return siguiente; }
   
   void SetSiguiente( nodo* sig )
   { siguiente = sig; }
   
  private:
   int numero;
   std::string nombre;
   nodo *siguiente;
};


Nota 5.

Código (cpp) [Seleccionar]
typedef nodo *pnodo; //definimos un tipo de dato personalizaodo. Tipo puntero de la clase NODO

Ya que usas un typedef... por qué no lo pones al principio y lo usas también en "nodo"??

Yo personalmente no soy partidario de usar alias para ahorrarte un asterisco... bueno, no te lo ahorras, lo cambias por una "p"... pero si lo usas, al menos intenta aprovecharlo al máximo:

Código (cpp) [Seleccionar]


class nodo; // declaration forward para el typedef

typedef nodo *pnodo; //definimos un tipo de dato personalizaodo. Tipo puntero de la clase NODO

class nodo {
  public:
   nodo(int num, const std::string& name, pnodo sig = NULL)
     : numero( num ),
       nombre( name ),
       siguiente( sig )
   {
   }//constructor
   
   int Numero( ) const
   { return numero; }
   
   std::string Nombre( ) const
   { return nombre; }
   
   pnodo Siguiente( ) const
   { return siguiente; }
   
   void SetSiguiente( pnodo sig )
   { siguiente = sig; }
   
  private:
   int numero;
   std::string nombre;
   pnodo siguiente;
};


Nota 6.

Código (cpp) [Seleccionar]
cola paciente;

No solo es muy mala idea usar variables globales sino que, además, la has metido entre medias de la implementación de "cola"... mi consejo es no usar variables globales, pero si las usas, al menos que estén en sitios concretos y correctamente identificadas.

Nota 7.

Como te comenté al principio, la STL tiene una colección bastante decente y completa de utilidades que te permiten "olvidar" las viejas funciones de C.

En el caso de acceso y manipulación de ficheros, tienes las clases ostream e istream ( y sus derivadas ) que simplifican la gestión de ficheros. Además vienen con opciones especiales para poder trabajar con clases nativas de C++, como la clase string.

Nota 8.

A la hora de guardar clases en ficheros no es buena idea guardar la clase "a pelo" volcando su memoria al fichero. La razón es que en ese volcado puede viajar información RTTI ( la que se usa en dynamic_cast y que gestiona la herencia )... y no creo que quieras exportar eso porque puedes tener problemas si cambia la estructura de las clases. La forma correcta de trabajar es guardar las cosas miembro a miembro. Lo mejor es crear una clase que sepa como "serializar" cada elemento "nodo" para almacenar / recuperar un nodo a partir de un archivo.

Por supuesto, lo que no puedes hacer es volcar el contenido de la cola literalmente... la cola almacena punteros... y lo que vas a almacenar en el fichero es el valor de los punteros, no el contenido apuntado por dichos punteros...

Nota 9.

Yo soy partidario de seguir la norma "1 clase = 1 responsabilidad"... simplifica el diseño, evita dependencias absurdas y facilita la depuración y el mantenimiento.

En tu caso, "cola" tiene 2 responsabilidades. Por un lado gestiona una lista de nodos y, por otra, permite volcar dichos nodos a la salida estándar... Yo habría optado por quitar esta segunda responsabilidad de "cola" y la habría puesto o bien en una clase creada para la ocasión o bien en una función independiente.

No hay que tener miedo a tener decenas de clases sencillas... al contrario de la creencia popular entre los que van aprendiendo, tener clases sencillas facilita muchísimo el proceso de crear software... es como usar piezas de Lego...

Si caes en la tentación de juntar múltiples responsabilidades en una sola clase o función al final acabarás gestionando clases gigantescas las cuales te van a devorar al intentar hacer modificaciones o correcciones... quedas avisado.