Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   Conexión con bases de datos (https://www.clubdelphi.com/foros/forumdisplay.php?f=2)
-   -   Creación de campos en runtime TClientdataset (https://www.clubdelphi.com/foros/showthread.php?t=88532)

fjcg02 18-06-2015 13:36:24

Creación de campos en runtime TClientdataset
 
Hola a todos,
estoy enfrascado en un problemilla y me gustaría saber qué opinais.

Tengo un TClientDataset al que le creo en runtime los campos que necesito. Hasta ahí bien, los creo, doy las altas de registros necesarios, preparo la información, y me funciona perfectamente. Como el Tclientdataset va a su bola, es decir, no extrae los datos de ninguna bbdd, ejecuto CDS.CreateDataset una vez creados los campos.

Sin embargo, cuando lanzo el proceso por segunda vez, cambiando los campos, me da un error. Concretamente cuand ejecuto CDS.CreateDataset. Previamente, he borrado los campos que había creado anteriormente.

También he intentado ejecutar el CreateDataset sólo la primera vez, pero también me da el error.

Sabría alguien decirme qué demonios se me está escapando ?

Gracias y un saludo

Así borro los campos:
Código Delphi [-]
for n:= CDSResultado.Fields.Count-1 downto 0 do
  CdSResultado.Fields[0].Free;
Así creo los campos:
Código Delphi [-]
function crearColumna(CDS:TClientDataset; Campo: string ):boolean;
var F: TStringField; NombreCampo:string;
begin
  F:= TStringField.Create(CDS);
  F.FieldName:= Campo;
  NombreCampo:= AnsiReplaceStr( Campo,' ', '_');
  NombreCampo:= trim(AnsiReplaceStr( NombreCampo,'-', '_'));
  F.Name:= CDS.Name+ NombreCampo;
  F.DataSet:= CDS;
  F.Size := 100;
  F.DisplayLabel:= Campo;
  F.DisplayWidth:= 10;

end;

Casimiro Noteví 18-06-2015 13:54:43

Olvido de novato: ¿qué error? :p

fjcg02 18-06-2015 14:15:17

Error:

CDS Field 'XX' not found


Obviamente, no debe existir, lo he borrado para volver a crear los campos.


Saludos

TOPX 18-06-2015 14:55:11

Hola,

Eso tiene que ver con la propiedad FieldDefs del TClientDataSet.

Más info en ~
¿Y si utiliza un nuevo ClientDataSet para la nueva estructura? En fin, son gustos.
-

Neftali [Germán.Estévez] 18-06-2015 15:03:25

¿No será algún otro componente que tienes "enganchado" a ese ClientDataset, quien está buscando el campo?

Caminante 18-06-2015 16:14:32

Hola

creo que deberia ser

Código Delphi [-]
Clientdataset1.FieldDefs.clear;

Al González 18-06-2015 17:19:22

Hola Javier.

Recuerda que puedes presionar Ctrl+C ("copiar") cuando te aparezca un mensaje de error. Así es fácil que nos muestres (Ctrl+V) el mensaje de error íntegro.

Saludos. :)

nlsgarcia 18-06-2015 17:28:42

fjcg02,

Cita:

Empezado por fjcg02
...TClientDataset al que le creo en runtime los campos que necesito...me funciona perfectamente...cuando lanzo el proceso por segunda vez, cambiando los campos, me da un error...

:rolleyes:

Revisa este código:
Código Delphi [-]
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls, DBClient, DB, Grids, DBGrids;

type
  TForm1 = class(TForm)
    Button1: TButton;
    Button2: TButton;
    Button3: TButton;
    Button4: TButton;
    Button5: TButton;
    DBGrid1: TDBGrid;
    procedure Button1Click(Sender: TObject);
    procedure Button2Click(Sender: TObject);
    procedure Button3Click(Sender: TObject);
    procedure Button4Click(Sender: TObject);
    procedure Button5Click(Sender: TObject);
    procedure FormClose(Sender: TObject; var Action: TCloseAction);
  private
    { Private declarations }
  public
    { Public declarations }
  end;

var
  Form1 : TForm1;
  Table : TClientDataset;
  DS : TDataSource;

implementation

{$R *.dfm}

// Crea campos del CDS
function CreateField(CDS : TClientDataset; FieldName : String) : Boolean;
var
   F : TStringField;

begin

   F := TStringField.Create(CDS);
   F.FieldName := FieldName;
   F.Name := FieldName;
   F.DataSet := CDS;
   F.Size := 100;
   F.DisplayLabel := 'N-' + FieldName;
   F.DisplayWidth := 10;

end;

// Crea el CDS
procedure TForm1.Button1Click(Sender: TObject);
begin
   if not Assigned(Table) then
      Table := TClientDataset.Create(nil);
end;

// Asigna campos al CDS
procedure TForm1.Button2Click(Sender: TObject);
begin
   if Assigned(Table) and (Table.State <> dsBrowse) then
   begin
      CreateField(Table,'C1');
      CreateField(Table,'C2');
      CreateField(Table,'C3');
      CreateField(Table,'C4');
      Table.CreateDataset;
   end;
end;

// Llena los campos del CDS
procedure TForm1.Button3Click(Sender: TObject);
var
   i : Integer;
begin
   if Assigned(Table) and (Table.FieldCount <> 0) then
   begin
     Randomize;
     Table.Open;
     Table.Append;
     for i := 0 to Table.FieldCount - 1 do
        Table.Fields[i].Value := 'Data-' + IntToStr(Random(1000));
     Table.Post;
   end;
end;

// Visualiza el CDS
procedure TForm1.Button4Click(Sender: TObject);
var
   i : Integer;
begin
   if Assigned(Table) then
   begin
      DS := TDataSource.Create(nil);
      DS.DataSet := Table;
      DBGrid1.DataSource := DS;
      for i := 0 to Table.FieldCount - 1 do
         DBGrid1.Columns[i].Width := 100;
   end;
end;

// Libera el CDS
procedure TForm1.Button5Click(Sender: TObject);
var
   i : Integer;

begin

   if Assigned(Table) and (Table.FieldCount <> 0) then
   begin
      DS.DataSet := nil;
      DBGrid1.DataSource := nil;
      for i := Table.Fields.Count - 1 to 0 do
         Table.Fields[i].Free;
      Table.Free;
      Table := nil;
   end;

   if Assigned(Table) then 
   begin
      Table.Free;
      Table := nil;
   end;

end;

// Libera recursos de Form1
procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
   Action := caFree;
end;

end.
El código anterior en Delphi 7 sobre Windows 7 Professional x32, Permite crear y eliminar un TClientDataset de forma dinámica como se muestra en la siguiente imagen:



Espero sea útil :)

Nelson.

fjcg02 18-06-2015 18:38:08

Hola amigos, gracias a todos por las respuestas.

Revisando un poco todo lo que habéis puesto, os apunto lo siguiente:

- El problema ocurre cuando cambio los campos que quiero visualizar. Si los campos son los mismos, no hay problema, se ejecuta perfectamente. Por lo tanto, el ejemplo de nlsgarcia, que es una pasada, en el que los campos son siempre los mismos, no reproduce mi problema.
- La solución que me propone Neftali, también la había probado, desactivando el dataset y el dbgrid del TClientDataset.
- Al, el mensaje que me da es el que he puesto, ni más, ni menos.

Finalmente, la sentencia mágica que ha hecho que funcione todo bien, es la que ha apuntado Caminante, demostrando que no hay camino, se hace camino al andar ;).

Código Delphi [-]
CDS.FieldDefs.Clear;


Por lo tanto , queda resuelto mi problema, ya sólo me queda poner edulcorantes al formulario para tener toda la artillería montada.


Lo dicho, muchas gracias a todos de nuevo y un saludo

ecfisa 18-06-2015 18:48:44

Hola fjcg02.

Fijate si este ejemplo se parece a lo que estas buscando,
Código Delphi [-]
// Crear columna
procedure CrearColumna(ClientDataSet: TClientDataSet; const FieldName: string;
  const FieldType: TFieldType; const FieldSize: Integer = 0);
begin
  with ClientDataSet do
  begin
    with FieldDefs.AddFieldDef do
    begin
      DataType := FieldType;
      Name     := FieldName;
      //...
      if FieldSize > 0 then Size := FieldSize;
    end;
  end;
end;

// Crear unas columnas y cargar (1)
procedure TForm1.Button1Click(Sender: TObject);
var
  i: Integer;
begin
  with ClientDataSet1 do
  begin
    Close;
    FieldDefs.Clear;
    CrearColumna(ClientDataSet1, 'Integer', ftInteger);
    CrearColumna(ClientDataSet1, 'String', ftString, 30);
    CrearColumna(ClientDataSet1, 'Float', ftFloat);
    CreateDataSet;
    Open;
    for i:= 1 to 1000 do
    begin
      Append;
      FieldByName('Integer').AsInteger := i;
      FieldByName('String').AsString   := 'Valor '+IntToStr(i);
      FieldByName('Float').AsFloat     := PI * i;
      Post;
    end;
    First;
  end;
end;

// Crear unas columnas y cargar (2)
procedure TForm1.Button2Click(Sender: TObject);
var
  i: Integer;
begin
  with ClientDataSet1 do
  begin
    Close;
    FieldDefs.Clear;
    CrearColumna(ClientDataSet1, 'String', ftString, 15);
    CrearColumna(ClientDataSet1, 'DateTime', ftDateTime);
    CrearColumna(ClientDataSet1, 'Currency', ftCurrency);
    CreateDataSet;
    Open;
    for i:= 1 to 1000 do
    begin
      Append;
      FieldByName('String').AsString     := 'Valor '+IntToStr(i);
      FieldByName('DateTime').AsDateTime := Now();
      FieldByName('Currency').AsFloat    := PI * i;
      Post;
    end;
    First;
  end;
end;

Saludos :)

Edito: Perdón fjcg02, no ví tu último mensaje mientras tenía este abierto. :o

fjcg02 18-06-2015 19:18:48

No importa Daniel, faltaría más, muchas gracias.

La verdad es que da gusto leer el código vuestro, no sé porqué siempre me parece mejor que el mío :)

También los links que ha propuesto TOPX me han sido bastante útiles, aunque volveré a echarles otro vistazo.

Un saludo

nlsgarcia 18-06-2015 19:22:34

fjcg02,

Cita:

Empezado por fjcg02
...la sentencia mágica que ha hecho que funcione todo bien, es la que ha apuntado Caminante (CDS.FieldDefs.Clear)...

^\||/

Cita:

Empezado por fjcg02
...El problema ocurre cuando cambio los campos que quiero visualizar...Si los campos son los mismos, no hay problema...el ejemplo de nlsgarcia...en el que los campos son siempre los mismos, no reproduce mi problema...

:rolleyes:

Revisa este código (Solo para complementar):
Código Delphi [-]
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls, DBClient, DB, Grids, DBGrids, Math;

type
  TForm1 = class(TForm)
    Button1: TButton;
    Button2: TButton;
    Button3: TButton;
    Button4: TButton;
    Button5: TButton;
    DBGrid1: TDBGrid;
    procedure Button1Click(Sender: TObject);
    procedure Button2Click(Sender: TObject);
    procedure Button3Click(Sender: TObject);
    procedure Button4Click(Sender: TObject);
    procedure Button5Click(Sender: TObject);
    procedure FormClose(Sender: TObject; var Action: TCloseAction);
  private
    { Private declarations }
  public
    { Public declarations }
  end;

var
  Form1: TForm1;
  Table : TClientDataset;
  DS : TDataSource;
  B : Boolean;

implementation

{$R *.dfm}

// Crea campos del CDS
function CreateField(CDS : TClientDataset; FieldName : String) : Boolean;
var
   F : TStringField;

begin

   F := TStringField.Create(CDS);
   F.FieldName := FieldName;
   F.Name := FieldName;
   F.DataSet := CDS;
   F.Size := 100;
   F.DisplayLabel := 'N-' + FieldName;
   F.DisplayWidth := 10;

end;

// Crea el CDS
procedure TForm1.Button1Click(Sender: TObject);
begin
   if not Assigned(Table) then
      Table := TClientDataset.Create(nil);
end;

// Asigna campos aleatorios al CDS
procedure TForm1.Button2Click(Sender: TObject);
begin

   Randomize;

   B := Not B;

   if Assigned(Table) and (Table.State <> dsBrowse) and B then
   begin
      CreateField(Table,'C' + IntToStr(RandomRange(10,19) + 1));
      CreateField(Table,'C' + IntToStr(RandomRange(20,29) + 1));
      CreateField(Table,'C' + IntToStr(RandomRange(30,39) + 1));
      CreateField(Table,'C' + IntToStr(RandomRange(40,49) + 1));
      Table.CreateDataset;
   end;

   if Assigned(Table) and (Table.State <> dsBrowse) and not B then
   begin
      CreateField(Table,'C' + IntToStr(RandomRange(50,59) + 1));
      CreateField(Table,'C' + IntToStr(RandomRange(60,69) + 1));
      CreateField(Table,'C' + IntToStr(RandomRange(70,79) + 1));
      CreateField(Table,'C' + IntToStr(RandomRange(80,89) + 1));
      Table.CreateDataset;
   end;

end;

// Llena los campos del CDS
procedure TForm1.Button3Click(Sender: TObject);
var
   i : Integer;
begin
   if Assigned(Table) and (Table.FieldCount <> 0) then
   begin
     Randomize;
     Table.Open;
     Table.Append;
     for i := 0 to Table.FieldCount - 1 do
        Table.Fields[i].Value := 'Data-' + IntToStr(Random(1000));
     Table.Post;
   end;
end;

// Visualiza CDS
procedure TForm1.Button4Click(Sender: TObject);
var
   i : Integer;
begin
   if Assigned(Table) then
   begin
      DS := TDataSource.Create(nil);
      DS.DataSet := Table;
      DBGrid1.DataSource := DS;
      for i := 0 to Table.FieldCount - 1 do
         DBGrid1.Columns[i].Width := 100;
   end;
end;

// Libera CDS
procedure TForm1.Button5Click(Sender: TObject);
var
   i : Integer;

begin

   if Assigned(Table) and (Table.FieldCount <> 0) then
   begin
      DS.DataSet := nil;
      DBGrid1.DataSource := nil;
      for i := Table.Fields.Count - 1 to 0 do
         Table.Fields[i].Free;
      Table.Free;
      Table := nil;
   end;

   if Assigned(Table) then 
   begin
      Table.Free;
      Table := nil;
   end;

end;

// Libera recursos de Form1
procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
   Action := caFree;
end;

end.
El código anterior en Delphi 7 sobre Windows 7 Professional x32, Permite crear y eliminar un TClientDataset con campos aleatorios de forma dinámica como se muestra en la siguiente imagen:



Nota: El ejemplo mostrado es básicamente similar al indicado en el Msg #8, la diferencia radica en que los campos creados son aleatorios.

Espero sea útil :)

Nelson.

fjcg02 18-06-2015 21:55:48

Gracias de nuevo, nlsgarcia.

Buen código, la diferencia que hay con el mío es que yo el TClientDataSet lo tengo ya creado en el formulario, y tú lo creas cada vez que pulsas el botón.

Puede ser una buena alternativa, ya que también me puede valer perfectamente. Lo voy a valorar.

Un saludo

ecfisa 18-06-2015 22:58:55

Hola fjcg02.

Otra situación que se te puede presentar, es agregar campos luego de haber creado el ClientDataSet. Para hacerlo sin perder los datos que llevas cargados, podes hacer como en este ejemplo:
Código Delphi [-]
procedure TForm1.Button2Click(Sender: TObject);
var
  cds : TClientDataSet;  // (me dan pereza los nombres largos)
  clon: TClientDataSet; // cds auxiliar
  i   : Integer;
begin
  cds  := ClientDataSet1;
  clon := TClientDataSet.Create(nil);
  try
    // Salvar datos anteriores
    clon.CloneCursor(cds, False, True );
    clon.First;
    // Agregar nuevos campos
    cds.Close;
    cds.FieldDefs.Update;
    cds.FieldDefs.Add('Campo3', ftCurrency, 0, False);
    cds.FieldDefs.Add('Campo4', ftFloat, 0, False);
    cds.CreateDataSet;
    cds.Open;
    // Recuperar los datos anteriores
    while not clon.Eof do
    begin
      cds.Append;
      cds.FieldByName('Campo1').AsInteger := clon.FieldByName('Campo1').AsInteger;
      cds.FieldByName('Campo2').AsString  := clon.FieldByName('Campo2').AsString;
      //...
      cds.Post;
      clon.Next;
    end;
    cds.First;
  finally
    clon.Free;
  end;
end;

Saludos :)

fjcg02 18-06-2015 23:25:08

Hola Daniel,
el caso que me planteas no se contempla en los casos de uso, ni creo que se dé.

El truco es bueno, clonar el clientdataset, añadir los campos y copiarlos luego.

Gracias, todas las ideas son buenas.

Finalmente he optado por la opción de destruir el clientdataset y volverlo a crear, pero tras hacerlo, ahora los datos que obtengo no son correctos, aunque como dicen los buenos moderadores:

"otro problema, otro hilo"

Ahora me voy a dormir, a ver si después de descansar encuentro el gazapo que antes no tenía.

Un saludo

Al González 18-06-2015 23:35:08

¡Joder, qué buenas aportaciones! El Club es grande. ||-||

fjcg02 19-06-2015 17:03:49

Cita:

Empezado por Al González (Mensaje 493455)
¡Joder, qué buenas aportaciones! El Club es grande. ||-||

A veces da miedo preguntar, te responden tanto y tan bueno, que luego estás haciendo deberes varios días.

Al final he hecho un mix, el datasource lo creo en diseño, y el el tclientdataset lo genero , libero y vuelvo a generar tantas veces como necesite. Funciona de maravilla.

A mis compañeros les ha gustado lo que he hecho: Sobre una consulta abierta, seleccionas campos para cabeceras de filas, campo para columnas, campo a utilizar, función de agregado ( count, sum, avg, max, min ) y voilá, te hace una pívot table en vivo y en directo que te muestra tan ricamente.

Gracias a todos y un saludo


La franja horaria es GMT +2. Ahora son las 09:45:24.

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