Club Delphi  
    FTP   CCD     Buscar   Trucos   Trabajo   Foros

Retroceder   Foros Club Delphi > Otros entornos y lenguajes > C++ Builder
Registrarse FAQ Miembros Calendario Guía de estilo Temas de Hoy

Respuesta
 
Herramientas Buscar en Tema Desplegado
  #1  
Antiguo 29-05-2007
GLGB GLGB is offline
Registrado
 
Registrado: may 2007
Posts: 8
Poder: 0
GLGB Va por buen camino
Question Manejo de archivo cvs

Hola que tal buenos días soy nueva en esto de Builder C++.... mi pregunta es la siguiente, tengo un archivo CVS el cual deseo pasar a una base de datos que ya tengo.

He utilizado lo siguiente:

1.- En un TMemo, paso el contenido de mi archivo separado por comas, y enseguida cada una de las lineas del Memo lo manejo como una cadena por separado donde utilizo un TStringList con el siguiente código


Código Delphi [-]

if (Memo->Lines->Count <=0)
ShowMessage ("Primero debe seleccionar un archivo");

else
{
TStringList *SL = new TStringList;

AnsiString cadena;

int i,contador=Memo->Lines->Count;

for (i=1; i<=contador; i++)
{
cadena=Memo->Lines->Strings[i];
SL->CommaText = cadena;
DM->Proveed->Insert();

int cantidad = SL->Count; // La cantidad de componentes

for (int fila=0; fila < cantidad; fila++)
{
Memo1->Lines->Add(SL->Strings[fila]);

if (fila==0)
DBEdit1->Text=SL->Strings[fila];
if (fila==2)
DBEdit2->Text=SL->Strings[fila];
if (fila==8)
DBEdit8->Text=SL->Strings[fila];
}
DM->Proveed->Post();
}
ShowMessage("ARCHIVO EXPORTADO.....");

}






Como se puede notar yo intente controlarlo con un If, y que cuando sea la segunda linea (nombre) me lo ponga en mi dbEdit pero el problema que tengo es que aquí no me separa por comas, si no por palabras, y si yo tengo el archivo con la siguiente estructura

CLAVE,NOMBRE,DIRECCION,TELEFONO
0001,GERARDO PEREZ GALINDO, GALEANA 211, 3949588
0020,AGUSTIN LOPEZ MENDOZA, MINA 344, 3948588

La separación sería:

001
GERARDO
PEREZ
GALINDO
GALEANA
211
3949588

y no puedo identificar cuántas líneas corresponden al nombre y cuantas a dirección, etc etc....

Espero alguien me pueda decir una mejor forma de realizar esto... muchas gracias
Responder Con Cita
  #2  
Antiguo 30-05-2007
Avatar de droguerman
droguerman droguerman is offline
Miembro
 
Registrado: abr 2005
Ubicación: tierra
Posts: 999
Poder: 20
droguerman Va por buen camino
Código:
if (Memo->Lines->Count <=0)
        ShowMessage ("Primero debe seleccionar un archivo");
 else
   {
        TStringList *SL = new TStringList;
    AnsiString cadena;
 
        int i,contador=Memo->Lines->Count;
 
        for (i=1; i<contador; i++)
         {
           cadena=Memo->Lines->Strings[i];
                  SL->CommaText = cadena;
                  DM->Proveed->Insert();
 
                  int cantidad = SL->Count; // La cantidad de componentes
 
                  for (int fila=0; fila < cantidad; fila++) 
               Memo1->Lines->Add(SL->Strings[fila]); 
 
         DBEdit1->Text=SL->Strings[0];
         DBEdit2->Text=SL->Strings[2];
         DBEdit8->Text=SL->Strings[8];
             }
             DM->Proveed->Post();
     }
     ShowMessage("ARCHIVO EXPORTADO.....");
 
   }
__________________
self.free;

Última edición por dec fecha: 30-05-2007 a las 08:46:54.
Responder Con Cita
  #3  
Antiguo 30-05-2007
Avatar de roman
roman roman is offline
Moderador
 
Registrado: may 2003
Ubicación: Ciudad de México
Posts: 20.269
Poder: 10
roman Es un diamante en brutoroman Es un diamante en brutoroman Es un diamante en bruto
Me parece GLGB que vas a tener que elaborar más. El problema es que la propiedad CommaText, como ya observaste, divide no sólo por comas sino también por espacios.

Lo ideal sería que los campos en el archivo CSV estuviesen delimitados, por ejemplo:

Código:
"0001","GERARDO PEREZ GALINDO", "GALEANA 211", "3949588"
"0020","AGUSTIN LOPEZ MENDOZA", "MINA 344", "3948588"
Pero si no tienes control sobre el formato del archivo, no se me ocure otra cosa que hacer el rastreo manualmente, esto es,

1. buscas la primera coma y apuntas su posición
2. copias la parte de la cadena desde el principio hasta esa posición
3. borras dela cadena la parte recién copiada
4. repites los pasos anteriores mientras la cadena no esté vacía.

Esta es la idea. No pongo un código más explícito porque desconozco las funciones en C que hacen eso, pero creo que puedes partir de ahí.

// Saludos
Responder Con Cita
  #4  
Antiguo 30-05-2007
GLGB GLGB is offline
Registrado
 
Registrado: may 2007
Posts: 8
Poder: 0
GLGB Va por buen camino
Red face

Hola.... pues si creo que si voy a tener que crear un algoritmo que vaya recorriendo uno a uno los caracteres....

Intente hacer lo siguiente:


Código Delphi [-]
if (Memo->Lines->Count <=0)
   ShowMessage ("Primero debe seleccionar un archivo");
else
   {
    TStringList *SL = new TStringList;
    AnsiString cadena;
    int i,contador_filas=Memo->Lines->Count;
    int no_caracteres;
        
    DM->Proveed->Insert();
    int no_filas;
    int caracter,flag=0;
    for(no_filas=1; no_filas<=contador_filas; no_filas++ )
       {
        flag=0;
        cadena=Memo->Lines->Strings[i]; // se asigna linea de texto del memo a la variable "cadena"
        no_caracteres=cadena.Length(); // Cuenta el numero de caracteres de la linea del Memo y se asigna a "no_caracteres"
        for ( caracter=0; caracter < no_caracteres; caracter++)
            {
             if (cadena[caracter]  !=  ',') // guarda los caracteres en Edit1 mientras sea diferente de ","
                  Edit1->Text=Edit1->Text + cadena[caracter];                          
             else      // Si en la posicion el caracter es una ","
                {
                 if(cadena[caracter+1]!=' ') // Si lo guardado en Edit1 es diferente de espacio en blanco
                   {                
                    flag++;
 
                    if(flag=1)
                      DBEdit1->Text=Edit1->Text;   
                 
                    if(flag=2)
                      DBEdit2->Text=Edit1->Text;
                    if(flag=3)
                      DBEdit8->Text=Edit1->Text;
                   
                    Edit1->Clear(); 
                   }            
            
                  else // Si la posicion despues de una "," es espacio en blanco...se incrementa "caracter" para empezar a guardar la siguiente cadena
                    caracter=caracter+2;
               }
               ShowMessage(Edit1->Text);
           
            } // For Caracter
      DM->Proveed->Post(); //Guarda en la base de datos los valores de DBEdits      
     } 
        
ShowMessage("ARCHIVO EXPORTADO.....");
}



Pero no me funciona, todo me lo pone en el DBEdit1, no se bien como hacer la separación de cadenas, alguien que me pueda echar la mano???
Responder Con Cita
  #5  
Antiguo 30-05-2007
GLGB GLGB is offline
Registrado
 
Registrado: may 2007
Posts: 8
Poder: 0
GLGB Va por buen camino
AAhh ahi en los ifs le tengo de la siguiente manera:


if (flag==1)
.......

if (flag==2)
.......

if (flag==3)
.......

osea q si le tengo doble == solo q en el anterior post se me paso ponerlo
Responder Con Cita
  #6  
Antiguo 30-05-2007
Avatar de roman
roman roman is offline
Moderador
 
Registrado: may 2003
Ubicación: Ciudad de México
Posts: 20.269
Poder: 10
roman Es un diamante en brutoroman Es un diamante en brutoroman Es un diamante en bruto
Algo que se me ocurre, un poco jalado de los pelos, pero que te permitiría usar la idea original, sería completar el CSV con los delimitadores conforme procesas cada línea

Código:
cadena = Memo->Lines->Strings;
Reemplazas todas las comas en cadena por la cadena "," y anexas el primero y último ". En delphi sería algo así:

Código Delphi [-]
cadena := Memo1.Lines[i];
cadena := '"' + StringReplace(cadena, ',', '"', [rfReplaceAll]) + '"';

Así, conviertes

0020,AGUSTIN LOPEZ MENDOZA, MINA 344, 3948588

en

"0020","AGUSTIN LOPEZ MENDOZA"," MINA 344"," 3948588"

con lo que puedes aplicar lo de CommaText sin problemas.

// Saludos
Responder Con Cita
  #7  
Antiguo 30-05-2007
GLGB GLGB is offline
Registrado
 
Registrado: may 2007
Posts: 8
Poder: 0
GLGB Va por buen camino
Entonces si se demilita por "1234","lucero","2332" el CommaText ya lo separará como yo quiero??Ya probé el código q me dice, pero me marca un error sintaxis y como yo nunca he usado esta función no se si me pudiera echar la mano...
Código Delphi [-]
cadena = '"' + StringReplace(cadena, ',', '"', ',', '"', [rfReplaceAll]) + '"';
Responder Con Cita
  #8  
Antiguo 30-05-2007
Avatar de roman
roman roman is offline
Moderador
 
Registrado: may 2003
Ubicación: Ciudad de México
Posts: 20.269
Poder: 10
roman Es un diamante en brutoroman Es un diamante en brutoroman Es un diamante en bruto
Así es, CommaText te separará ahora sólo por las comas. Hay un error en el código que puse. Debería ser

Código Delphi [-]
cadena := '"' + StringReplace(cadena, ',', '","', [rfReplaceAll]) + '"';

en lugar de

Código Delphi [-]
cadena := '"' + StringReplace(cadena, ',', '"', [rfReplaceAll]) + '"';

En C habrá que ajustar porque las " se usan para las cadenas a diferencia de delphi que usa el apóstrofe '.

// Saludos
Responder Con Cita
  #9  
Antiguo 30-05-2007
Avatar de delphi.com.ar
delphi.com.ar delphi.com.ar is offline
Federico Firenze
 
Registrado: may 2003
Ubicación: Buenos Aires, Argentina *
Posts: 5.933
Poder: 27
delphi.com.ar Va por buen camino
No se olviden de los siguientes detalles:
  • Puede haber comas dentro de los valores, es decir encerradas entre comillas y hay que tomarlas como parte del dato
  • Puede haber comillas dentro de los valores, al igual que las comas hay que tomarlas como datos. Dos comillas seguidas indican una comilla de dato.


Mi componente FileReader, tiene la lógica para "desmenuzar" un CSV. Nunca he publicado el código porque fue de las primeras cosas que hice en Delphi y creo que si lo veo ahora debe ser vergonzoso. Si estoy seguro que funciona sin problemas, ya que lo he utilizado en infinidad de aplicaciones.

Saludos!
__________________
delphi.com.ar

Dedique el tiempo suficiente para formular su pregunta si pretende que alguien dedique su tiempo en contestarla.
Responder Con Cita
  #10  
Antiguo 30-05-2007
Avatar de roman
roman roman is offline
Moderador
 
Registrado: may 2003
Ubicación: Ciudad de México
Posts: 20.269
Poder: 10
roman Es un diamante en brutoroman Es un diamante en brutoroman Es un diamante en bruto
Había pensado en eso, pero ¡qué exigentes eres!

La verdad es que tienes razón y habría que cubrir todos los casos. La componente que mencionas, ¿vas a publicar su código? Porque habría que traducirlo a C.

// Saludos
Responder Con Cita
  #11  
Antiguo 30-05-2007
Avatar de seoane
[seoane] seoane is offline
Miembro Premium
 
Registrado: feb 2004
Ubicación: A Coruña, España
Posts: 3.717
Poder: 24
seoane Va por buen camino
Bueno, no creo que a los de borland les importe mucho que partiendo del método SetDelimitedText y recortando aquí y allá, lleguemos a esto:
Código Delphi [-]
procedure SetDelimitedText(StringList: TStringList; const Str: string);
var
  P, P1: PChar;
  S: string;
begin
  with StringList do
  begin
    BeginUpdate;
    try
      Clear;
      P := PChar(Str);
      while P^ <> #0 do
      begin
        if P^ = QuoteChar then
          S := AnsiExtractQuotedStr(P, QuoteChar)
        else
        begin
          P1 := P;
          while (P^ <> #0) and (P^ <> Delimiter) do
            P := CharNext(P);
          SetString(S, P1, P - P1);
        end;
        Add(S);
        if P^ = Delimiter then
        begin
          P:= CharNext(P);
          if P^ = #0 then
            Add('');
        end;
      end;
    finally
      EndUpdate;
    end;
  end;
end;

Por ejemplo:
Código Delphi [-]
var
  Lista: TStringList;
begin
  Lista:= TStringList.Create;
  try
    SetDelimitedText(Lista,'0001,GERARDO PEREZ GALINDO, GALEANA 211, 3949588');
    ShowMessage(Lista.Text);
  finally
    Lista.Free;
  end;
end;
Seguimos teniendo que traducirlo a C, pero teniendo en cuenta que se hace un uso intensivo de punteros, creo que el paso a C tendría que ser sencillo.

PD: Creo que ya lo dije antes, pero en el nuevo turbo la propiedad StrictDelimiter soluciona este problema.

Última edición por seoane fecha: 30-05-2007 a las 19:46:40.
Responder Con Cita
  #12  
Antiguo 30-05-2007
Avatar de roman
roman roman is offline
Moderador
 
Registrado: may 2003
Ubicación: Ciudad de México
Posts: 20.269
Poder: 10
roman Es un diamante en brutoroman Es un diamante en brutoroman Es un diamante en bruto
Cita:
Empezado por seoane
Bueno, no creo que a los de borland les importe mucho que partiendo del método SetDilimitedText y recortando aquí y allá, lleguemos a esto
Seguro que no, ya Andreano dijo que la VLC era Open Source.



// Saludos
Responder Con Cita
  #13  
Antiguo 30-05-2007
Avatar de roman
roman roman is offline
Moderador
 
Registrado: may 2003
Ubicación: Ciudad de México
Posts: 20.269
Poder: 10
roman Es un diamante en brutoroman Es un diamante en brutoroman Es un diamante en bruto
Por cierto, donde dices SetDilimitedText, ¿no quieres decir SetDelimitedText?



// Saludos
Responder Con Cita
  #14  
Antiguo 30-05-2007
Avatar de seoane
[seoane] seoane is offline
Miembro Premium
 
Registrado: feb 2004
Ubicación: A Coruña, España
Posts: 3.717
Poder: 24
seoane Va por buen camino
Cita:
Empezado por roman
Por cierto, donde dices SetDilimitedText, ¿no quieres decir SetDelimitedText?
Veo que tienes alma de compilador
Responder Con Cita
  #15  
Antiguo 31-05-2007
GLGB GLGB is offline
Registrado
 
Registrado: may 2007
Posts: 8
Poder: 0
GLGB Va por buen camino
Smile

Gracias por sus sugerencias, de hecho se me hizo un poco complicado conl a solución que me dieron, además que seguí trabajando con el código que ya tenía. Ya me función y está en Builder, x si a alguien le sirve

Código Delphi [-]

if (Memo->Lines->Count <=0)
   ShowMessage ("Primero debe seleccionar un archivo");
else
   {
    AnsiString cadena;
    int contador_filas;
    int no_caracteres;
    int no_filas;
    int caracter,flag;
    
    contador_filas=Memo->Lines->Count;
    
    for(no_filas=1; no_filas<=contador_filas; no_filas++ )
       {
        flag=0;
        cadena=Memo->Lines->Strings[no_filas];           
        
        no_caracteres=cadena.Length(); 
 
       for ( caracter=1; caracter<= no_caracteres; caracter++)
            {
             if (cadena[caracter] != ',')
                  Edit1->Text=Edit1->Text + cadena[caracter];
             
              else                   
                 {
                    flag=flag+1;
 
                    if(flag==1)
                      Edit2->Text=Edit1->Text;
 
                     if (flag==2)
                        Edit5->Text=Edit1->Text;
 
                     if(flag==3)
                      Edit3->Text=Edit1->Text;
 
                     if(flag==5)
                      Edit4->Text=Edit1->Text;
 
                    Edit1->Clear();
               }
     }
ShowMessage("ARCHIVO EXPORTADO.....");
}


Gracias a todos!!! y espero les sirva de algo mi código....
Responder Con Cita
  #16  
Antiguo 31-05-2007
Avatar de seoane
[seoane] seoane is offline
Miembro Premium
 
Registrado: feb 2004
Ubicación: A Coruña, España
Posts: 3.717
Poder: 24
seoane Va por buen camino
Bonita solución GLGB, aunque debes de tener cuidado de que no aparezcan comas en el medio de uno de los campos. De todas formas si estas seguro de que eso nunca va a ocurrir, la solución parece eficaz.

Por cierto, ya que hablamos de cvs, hace poco puse en mi pagina web un pequeño código para manipular ese tipo de ficheros. Permite modificar el delimitador, las comillas, el número de campos y su orden.

http://delphi.jmrds.com/?q=node/28

Por si a alguien le interesa ...
Responder Con Cita
  #17  
Antiguo 31-05-2007
GLGB GLGB is offline
Registrado
 
Registrado: may 2007
Posts: 8
Poder: 0
GLGB Va por buen camino
Gracias.. lo tomare en cuenta por si algo falla con mi algoritmo!!

Saludos
Responder Con Cita
Respuesta



Normas de Publicación
no Puedes crear nuevos temas
no Puedes responder a temas
no Puedes adjuntar archivos
no Puedes editar tus mensajes

El código vB está habilitado
Las caritas están habilitado
Código [IMG] está habilitado
Código HTML está deshabilitado
Saltar a Foro

Temas Similares
Tema Autor Foro Respuestas Último mensaje
Filosofia/Manejo archivo "cds" Archivo.cds!! Martín González Conexión con bases de datos 1 03-05-2007 00:14:23
Manejo de una DLL guiweb Varios 4 30-06-2006 15:10:48
Manejo de Archivo, Consulta?? marceloalegre Varios 2 07-11-2005 15:29:40
manejo de BD raco Varios 0 21-07-2005 00:25:33
manejo de .dbf en delphi jeni Conexión con bases de datos 2 14-09-2003 02:09:42


La franja horaria es GMT +2. Ahora son las 05:19:28.


Powered by vBulletin® Version 3.6.8
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Traducción al castellano por el equipo de moderadores del Club Delphi
Copyright 1996-2007 Club Delphi