Club Delphi  
    FTP   CCD     Buscar   Trucos   Trabajo   Foros

Retroceder   Foros Club Delphi > Bases de datos > Firebird e Interbase
Registrarse FAQ Miembros Calendario Guía de estilo Temas de Hoy

Respuesta
 
Herramientas Buscar en Tema Desplegado
  #1  
Antiguo 04-07-2007
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.556
Poder: 25
egostar Va camino a la fama
Relación de tablas con llaves foráneas

Hola amigos,

Espero poder ser claro con algunas dudas que tengo.

Estoy creando una relación de tablas usando llaves foráneas.

Uso Firebird 2.0 y componentes IBX en Turbo Delphi

Las tablas que tengo, sus llaves primarias (PK) y sus llaves foráneas (FK) son las siguientes:

RECTORIAS


PK ID_Rectoria VARCHAR(3)
Nombre VARCHAR(30)

DIVISIONES

FK PK ID_Rectoria VARCHAR(3)
PK ID_Division VARCHAR(3)
Nombre VARCHAR(30)

DEPARTAMENTOS

FK PK ID_Rectoria VARCHAR(3)
FK PK ID_Division VARCHAR(3)
PK ID_Departamento VARCHAR(3)
Nombre VARCHAR(30)

SECCIONES

FK PK ID_Rectoria VARCHAR(3)
FK PK ID_Division VARCHAR(3)
FK PK ID_Departamento VARCHAR(3)
PK ID_Seccion VARCHAR(3)
Nombre VARCHAR(30)

Intento con esta relación de tablas generar agrupaciones de tipo organizacional.

Nota: Uso VarChar(3) ya que requiero que los datos tengan la forma:

RECTORIA+DIVISION+DEPARTAMENTO+SECCION
001+001+001+001

Preguntas:

¿Están bien creadas las relaciones y llaves?

¿Como generar automáticamente un consecutivo 001-002-003 en los campos dentro de la base de datos (con un procedimiento almacenado o con un Disparador)? Por ejemplo:

Cuando presiono el boton de Agregar en RECTORIA

RECTORIA = 001+

Cuando agrego una DIVISION y selecciono una RECTORIA, genere el siguiente número para la DIVISION

DIVISION = RECTORIA 001+

Cuando agrego una DEPARTAMENTO y selecciono una RECTORIA y una DIVISION, genere el siguiente número para el DEPARTAMENTO

DEPARTAMENTO = RECTORIA DIVISION 001+

Cuando agrego una SECCION y selecciono una RECTORIA una DIVISION y un DEPARTAMENTO, genere el siguiente número para el DEPARTAMENTO

SECCION = RECTORIA DIVISION DEPARTAMENTO 001+

Espero haber sido claro con mis desorden mental.

Gracias anticipadas.

Salud OS.
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney
Responder Con Cita
  #2  
Antiguo 04-07-2007
gabrielkc gabrielkc is offline
Miembro
 
Registrado: jun 2007
Ubicación: Chihuahua Mexico
Posts: 118
Poder: 17
gabrielkc Va por buen camino
Una rectoría tiene un valor único, cuando creas una division se crea con un valor único le agregas la llave de la rectoría a la que pertenece hasta ahi va bien.

El problema es cuando creas un departamento

DEPARTAMENTO = RECTORIA DIVISION 001+

el departamento tiene un valor único el cual debes relacionar con una división, la rectoría en este caso sobra dado que sabiendo la division puedes saber la Rectoría, estas redundando datos.

Por lo tanto haces lo mismo cuando creas una división

SECCION = RECTORIA DIVISION DEPARTAMENTO 001+


con el valor del departamento al que pertenece, puedes obtener la división y con esta la rectoría.
Responder Con Cita
  #3  
Antiguo 04-07-2007
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.556
Poder: 25
egostar Va camino a la fama
Si, de hecho eso mismo hago ahora mismo, pero debo capturar el valor del concepto correspondiente

RECTORIA
DIVISION
DEPARTAMENTO
SECCION


El problema que le veo es que la captura requiere de saber cual es el último dato ingresado y no quiero que sea un campo autoincrementable (cosa que creo no puede ser porque son campos VARCHAR).

Aqui muestro el bosquejo del módulo que estoy creando.



En realidad lo único que quiero que se capture es el nombre de la SECCION en este caso.

Salud OS.

Edito: Además, quiero que esto lo haga la base de datos usando un Procedimiento almacenado o un Disparador, bueno, así lo he soñado, no quisiera hacerlo por código, que es otra alternativa si no hay una solución a esta duda.
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney

Última edición por egostar fecha: 04-07-2007 a las 22:28:16.
Responder Con Cita
  #4  
Antiguo 04-07-2007
gabrielkc gabrielkc is offline
Miembro
 
Registrado: jun 2007
Ubicación: Chihuahua Mexico
Posts: 118
Poder: 17
gabrielkc Va por buen camino
mmm y si llamas un query con

SELECT MAX(ID_RECTORIA),NOMBRE FROM RECTORIA GROUP BY NOMBRE

y luego en el combo (en el cual supongo cargas los nombres de las rectorias, divisiones y departamentos), haces:

ComboRectoria.ItemIndex:=ComboRectoria.Items.IndexOf(Query.FieldByName('NOMBRE').AsString));

y obviamente lo mismo para división y departamento???
Responder Con Cita
  #5  
Antiguo 04-07-2007
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.556
Poder: 25
egostar Va camino a la fama
Si gracias, con código es una opción que tengo contemplada como última opción.

He creado el siguiente procedimiento almacenado:

Código SQL [-]
begin
  /* Procedure Text */
  FOR SELECT seccion FROM SECCIONES
     WHERE RECTORIA = :c_rectoria and DIVISION = :c_division and DEPARTAMENTO = :c_depto
     INTO :c_seccion DO
  begin
    suspend;
  end
end

Si me funciona, pero me da el primer dato, lo que quiero es si es posible que dentro de ese procedimiento obtener el último valor para esa comparación (where .....)

Gracias por la ayuda.

Salud OS.
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney
Responder Con Cita
  #6  
Antiguo 04-07-2007
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.556
Poder: 25
egostar Va camino a la fama
Bueno, usando tu Query hice esto en el procedimiento.

Código SQL [-]
begin
  /* Procedure Text */
  FOR SELECT max(seccion) FROM SECCIONES
     WHERE RECTORIA = :c_rectoria and DIVISION = :c_division and DEPARTAMENTO = :c_depto
     INTO :c_seccion DO
  begin
    suspend;
  end
end
Efectivamente me regresa el último dato.

Ahora solo es ver de que manera lo coloco en el campo de mi forma con el siguiente valor.

Muchas gracias.

Salud OS.
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney
Responder Con Cita
  #7  
Antiguo 04-07-2007
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.556
Poder: 25
egostar Va camino a la fama
Solo para complementar este hilo les dejo el complemento de código que me funcionó como queria.

Al seleccionar el departamento, paso los parametros al SP, calculo el siguiente valor y le doy el formato deseado '000'.

Código Delphi [-]
procedure TFEstructura.DBLookupComboBox6CloseUp(Sender: TObject);
var
  Seccion : string;
  SeccionInt: integer;
begin
  with DMEstructura do begin
     IBSPSecciones.Prepare;
     IBSPSecciones.ParamByName('C_RECTORIA').Value :=  IBRectorias.FieldByName('RECTORIA').Value;
     IBSPSecciones.ParamByName('C_DIVISION').Value := IBDivisiones.FieldByName('DIVISION').Value;
     IBSPSecciones.ParamByName('C_DEPTO').Value := IBDeptos.FieldByName('DEPARTAMENTO').Value;
     IBSPSecciones.ExecProc;
     //Valido si se encontró algún dato en caso contrario inicializo en cero.
     if IBSPSecciones.Params[0].Value <> null then 
        Seccion := IBSPSecciones.Params[0].Value
     else Seccion := '0';
     IBSPSecciones.UnPrepare;
     SeccionInt := StrtoInt(Seccion)+1;
     //Asigno el formato 00x a partir del valor entero.
     IBSecciones.FieldByName('SECCION').Value := Format('%.*d',[3,SeccionInt]);
  end;
  DBMemo4.SetFocus;
end;

No se si lo que hago esta bien o sea lo correcto, pero he visto en varios hilos que es bueno que quien haga el trabajo sea la base de datos, bueno, para eso me imagino existen los SP.

Espero que a alguien le sirva.

Salud OS.
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney
Responder Con Cita
  #8  
Antiguo 05-07-2007
Avatar de Ivanzinho
[Ivanzinho] Ivanzinho is offline
Miembro Premium
 
Registrado: ene 2005
Ubicación: Galicia
Posts: 595
Poder: 20
Ivanzinho Va por buen camino
Si lo quieres hacer todo dentro de un procedimiento prueba lo siguiente :
Código SQL [-]
begin
  /* Procedure Text */
  //Obtenemos el campo sección como un entero
  FOR SELECT Cast(max(seccion) as integer) FROM SECCIONES
     WHERE RECTORIA = :c_rectoria and DIVISION = :c_division and DEPARTAMENTO = :c_depto
     INTO :c_seccionint DO
  begin
    //Si es nulo lo inicializamos a 0
    if (c_seccionint is null) then
      c_seccion = 0;
    
    //Aumentamos el contador
    c_seccionint = c_seccionint + 1;

    //Pasamos el campo a VarChar(3)
    c_seccion = Cast(c_seccionint as VarChar(3))

    //Comprobamos que el campo tenga siempre tres caracteres,
    //esto lo pongo porque creo que quieres que el campo tenga
    //el formato 001 y no 1. Si te vale la forma 1 olvidate del while.
    while length(c_seccion) < 3 do
       c_seccion = '0' || c_seccion;

    //Para finalizar con el campo c_seccion ejecutas la acción que
    //desees, un insert, update, ...
   
  end
end

Espero que te sirva

Un saúdo.
__________________
Si no lees esto no vivirás tranquilo
Non lle poñades chatas â obra namentras non se remata. O que pense que vai mal que traballe n’ela; hai sitio para todos. (Castelao)

Última edición por Ivanzinho fecha: 05-07-2007 a las 09:17:38.
Responder Con Cita
  #9  
Antiguo 05-07-2007
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.556
Poder: 25
egostar Va camino a la fama
Muchas gracias Ivanzinho, lo voy a probar y comento los resultados.

Salud OS.
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney
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
LLaves foraneas... Luis Castillo SQL 2 13-11-2005 18:45:34
Problema de llaves primarias y foráneas!! romansiux Varios 2 10-04-2005 16:20:57
Llaves Foraneas RainFall MySQL 1 26-07-2004 04:19:28
Llaves foraneas en BDD distintas StartKill Firebird e Interbase 7 31-01-2004 01:14:01
Borrar una tabla que tiene llaves foraneas afarenas Firebird e Interbase 2 10-10-2003 22:00:15


La franja horaria es GMT +2. Ahora son las 15:16:54.


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