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 10-12-2010
Toni Toni is offline
Miembro
 
Registrado: may 2003
Ubicación: Barcelona - España
Posts: 364
Poder: 22
Toni Va por buen camino
Problemas con ClientDataSets dinamicos

Hola,

Desarrollo las aplicaciones utilizando BCB6 + FB1.5 y utilizo para acceder a los datos ClientDataSet + DatasetProvider + IBQuery.

En algunas ocasiones al realizar el ApplyUpdates al ClientDataSet me genera el evento OnReconcileError y me da el error 'record not found'.

Este error sucede porque no puede encontrar el registro que tiene que actualizar y esto a su vez porque el ClientDataSet no ha podido averiguar cual es la clave primario de la tabla.

Una solucion al problema es crear los campos de forma estatica en el IBQuery y marcar que los campos que son la clave primaria mediante la propiedad ProviderFlags->pfInKey.

Como decia al principio este error no me sucede siempre y me gustaria poder continuar sin utilizar los campos estaticos. Por cuestiones de que a medida que voy modificando las estructuras de las tablas no tengo porque modificar la aplicación y añadir los campos nuevos como estaticos para poder acceder a los nuevos campos.

El problema es que cuando no se utilizan los campos estaticos, pues no sabe cual es la clave principal y da problemas para actualizar (curiosamente pocas veces)

Haber si alguien tiene alguna sugerencia al respecto.
__________________
Saludos,

Bitman
Responder Con Cita
  #2  
Antiguo 10-12-2010
Avatar de Al González
[Al González] Al González is offline
In .pas since 1991
 
Registrado: may 2003
Posts: 5.604
Poder: 30
Al González Es un diamante en brutoAl González Es un diamante en brutoAl González Es un diamante en brutoAl González Es un diamante en bruto
Hola.

Algo como lo siguiente hago yo para todos los conjuntos de datos a los que asocio un TDataSetProvider:

Código Delphi [-]
Var
  Field :TField;
Begin
  Field := DataSet.FindField ('ID');

  If Field <> Nil Then
    // Activamos la bandera pfInKey en el campo de llave primaria
    Field.ProviderFlags := Field.ProviderFlags + [pfInKey];

Suponiendo que no uses un nombre común para el campo llave (hay a quienes les gusta la incómoda repetición del nombre de la tabla como parte del nombre de su campo principal, y también quienes desestiman las ventajas de las llaves artificiales), entonces puedes tomar el primer campo, asumiendo que éste siempre será el campo de llave primaria:

Código Delphi [-]
  Field := DataSet.Fields [0];

(DataSet es tu IBQuery)

Esto lo puedes hacer en algún evento o punto del programa donde ya estén creados los objetos TField de un IBQuery, pero antes de que el primer paquete de datos sea solicitado por el conjunto de datos cliente correspondiente.

O también puedes redefinir el método virtual TDataSet.PSGetKeyFields en una clase derivada de TIBQuery.

Como suelo usar campos persistentes, no me he visto en la necesidad de estudiar a fondo alguno de esos caminos, pero conociendo al señor Delphi, no dudo que sea posible lograrlo a través de esas u otras ideas de solución.

No dejes de retroalimentar este tema.

Saludos.

Al González.
Responder Con Cita
  #3  
Antiguo 11-12-2010
Toni Toni is offline
Miembro
 
Registrado: may 2003
Ubicación: Barcelona - España
Posts: 364
Poder: 22
Toni Va por buen camino
Hola Al Gonzalez,

Si ya contaba con esta posibilidad de asignar mediante codigo en cada IBQuery la clave principal. Lo que me preguntaba si Delphi/Builder no lo hacia automaticamente de alguna manera. En mi caso cada tabla es totalmente diferente algunas tienen claves de varios campos y los campos estan identificados con nombre diferentes. Seguramente optare por asignar mediante codigo en cada IBQuery los campos de la clave principal, porque la clave principal no suele cambiar y asi me permite acceder a cualquier campo nuevo que añada en las tablas sin tener que recompilar la aplicacion. La otra opcion que es crear los campos estaticos en los IBQuery no me gusta mucho porque cualquier modificacion en las tablas requiere crearlos de nuevo y recompilar.

Muchas gracias por tu sugerencia.
__________________
Saludos,

Bitman
Responder Con Cita
  #4  
Antiguo 13-12-2010
Toni Toni is offline
Miembro
 
Registrado: may 2003
Ubicación: Barcelona - España
Posts: 364
Poder: 22
Toni Va por buen camino
Finalmente lo he solucionado mediante codigo poniendo en cada IBQuuery en el metodo AfterOpen, ya que no he encontrado ninguna forma de que me detecte automaticamente los campos que son la llave primaria.


Código Delphi [-]
void __fastcall TfrmDocumento::qryrw_CabOrdReubicacionAfterOpen(TDataSet *DataSet)
{
  DataSet->FieldByName("idEmpresa")->ProviderFlags << pfInKey;
  DataSet->FieldByName("idEjercicio")->ProviderFlags << pfInKey;
  DataSet->FieldByName("idSerie")->ProviderFlags << pfInKey;
  DataSet->FieldByName("NumeroDocumento")->ProviderFlags << pfInKey;
}
__________________
Saludos,

Bitman
Responder Con Cita
  #5  
Antiguo 13-12-2010
Avatar de fjcg02
[fjcg02] fjcg02 is offline
Miembro Premium
 
Registrado: dic 2003
Ubicación: Zamudio
Posts: 1.410
Poder: 22
fjcg02 Va camino a la fama
Hola,

en lugar de hacerlo 'a mano', podrías preguntar al motor de la base de datos qué campos son PK, y luego ejecutar el código que te propone Al. Haciendo una función, podrías llamarla para cada TDataset antes de abrirlo. Seguro que te simplifica el código.

Un saludo
__________________
Cuando los grillos cantan, es que es de noche - viejo proverbio chino -
Responder Con Cita
  #6  
Antiguo 13-12-2010
Toni Toni is offline
Miembro
 
Registrado: may 2003
Ubicación: Barcelona - España
Posts: 364
Poder: 22
Toni Va por buen camino
Eso seria perfecto, de hecho es lo que pretendia preguntar, pero como averiguo que campos son la clave principal de una base de datos que existe y las claves son 1 o mas campos?
__________________
Saludos,

Bitman
Responder Con Cita
  #7  
Antiguo 13-12-2010
Avatar de fjcg02
[fjcg02] fjcg02 is offline
Miembro Premium
 
Registrado: dic 2003
Ubicación: Zamudio
Posts: 1.410
Poder: 22
fjcg02 Va camino a la fama
Prueba con esto

Código SQL [-]
  select  i.rdb$field_name
   from rdb$relation_constraints rc, rdb$index_segments i, rdb$indices idx
   where i.rdb$index_name = rc.rdb$index_name and
         idx.rdb$index_name = rc.rdb$index_name and
         rc.rdb$constraint_type = 'PRIMARY KEY' and
         rc.rdb$relation_name = :TABLA

Cortesía del compañero Al, de una bbdd que colgó en su página web GHSistemas y que ya no está operativa.
En la página de firebird también tienes información al respecto.

Ya nos dirás.

Saludos
__________________
Cuando los grillos cantan, es que es de noche - viejo proverbio chino -
Responder Con Cita
  #8  
Antiguo 14-12-2010
Toni Toni is offline
Miembro
 
Registrado: may 2003
Ubicación: Barcelona - España
Posts: 364
Poder: 22
Toni Va por buen camino
Fantastico! Voy a probar lo os comento.
__________________
Saludos,

Bitman
Responder Con Cita
  #9  
Antiguo 16-12-2010
Toni Toni is offline
Miembro
 
Registrado: may 2003
Ubicación: Barcelona - España
Posts: 364
Poder: 22
Toni Va por buen camino
Hola a todos,

Por el momento lo tengo solucionado llamando desde el evento AfterOpen del IBQuery a la funcion AsignarCamposClavePrincipal:

Código Delphi [-]
void __fastcall TDM2::AsignarCamposClavePrincipal(TDataSet *DataSet, AnsiString Tabla)
{
 rdb_indices->Close();
 rdb_indices->ParamByName("P_TABLA")->AsString = Tabla;
 rdb_indices->Open();
 rdb_indices->First();
 while(!rdb_indices->Eof)
  {
    DataSet->FieldByName(rdb_indices->FieldByName("rdb$field_name")->AsString.Trim())->ProviderFlags <Next();
  }
 rdb_indices->Close();
}

Este es codigo SQL del IBquery rdb_indices que se utiliza en la funcion:

Código SQL [-]
select  i.rdb$field_name
   from rdb$relation_constraints rc, rdb$index_segments i, rdb$indices idx
   where i.rdb$index_name = rc.rdb$index_name and
         idx.rdb$index_name = rc.rdb$index_name and
         rc.rdb$constraint_type = 'PRIMARY KEY' and
         rc.rdb$relation_name = :P_TABLA
__________________
Saludos,

Bitman
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
dbExpress - ClientDataSets anidados PabloZZZ Conexión con bases de datos 4 13-10-2010 22:58:29
ClientDataSets y Firebird Walterdf Conexión con bases de datos 19 27-08-2010 20:41:31
Capturar errores - ClientDataSets rochi Providers 3 22-11-2008 00:05:17
Clientdatasets anidados con ADO Johnny Q Conexión con bases de datos 4 03-11-2005 02:53:25
Problemas con componentes dinámicos mamaro OOP 2 14-10-2004 17:14:06


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


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