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 02-09-2011
Bretema Bretema is offline
Miembro
 
Registrado: nov 2003
Ubicación: Vigo - Galicia
Posts: 37
Poder: 0
Bretema Va por buen camino
¿Lentitud en proceso de registros ...?

Buenos días,

tengo una aplicación que maneja una base de datos a través de Firebird con componentes IBX. Uno de los procesos consiste en la elaboración de un diario con registros importados de varias tablas. Una vez tengo la tabla del diario cargada con sus movimientos, leo de otra tabla el saldo inicial de la cuenta y después recorro todos los movimientos del diario estableciendo el saldo después de cada movimiento. El código es simple:


Código:
 TMov.Open;
 TMov.First;

 while not TMov.Eof do
    begin
    TMov.Edit;

    TMovSaldo.Value := Saldo + TMovDebe.Value - TMovHaber.Value;
    Saldo := TMovSaldo.Value;
    DebeMo := DebeMo + TMovDebe.Value;
    HaberMo := HaberMo + TMovHaber.Value;

    TMov.Post;
    TMov.Next;
    end;
El problema viene dado que, ahora que tengo diarios con mas de 2.000 movimientos, el recorrer los registros para incluir el saldo tarda mucho, estamos hablando de cerca de 30 segundos con un Pentium 4 ..... o unos 10 en un dualcore.

El código, desde mi modesto punto de vista, no tiene mucho mas donde rascar .... pero me parece excesivo el tiempo de procesamiento para ese numero de registros ....

He hecho la prueba exportando las tablas a Interbase y a traves de ese gestor los tiempos empeoran ligeramente.

Alguna sugerencia ..... ?
Responder Con Cita
  #2  
Antiguo 02-09-2011
Avatar de guillotmarc
guillotmarc guillotmarc is offline
Miembro
 
Registrado: may 2003
Ubicación: Huelva
Posts: 2.638
Poder: 24
guillotmarc Va por buen camino
Esto es un ejemplo perfecto de un proceso que se ejecuta muchísimo más rápido en un procedimiento almacenado.

Sería más o menos así :

Código SQL [-]
SET TERM ^ ;

create procedure ACTUALIZAR_SALDO
returns (
    TOTAL_SALDO numeric(15,2),
    TOTAL_DEBE numeric(15,2),
    TOTAL_HABER numeric(15,2))
as
declare variable MOV_ID integer;
declare variable MOV_DEBE numeric(15,2);
declare variable MOV_HABER numeric(15,2);
begin
  TOTAL_SALDO = 0;
  TOTAL_DEBE = 0;
  TOTAL_HABER = 0;

  for select ID, coalesce(DEBE,0), coalesce(HABER,0)
      from MOVIMIENTOS
      into :MOV_ID, :MOV_DEBE, :MOV_HABER
  do begin
    TOTAL_SALDO = TOTAL_SALDO + MOV_DEBE - MOV_HABER;
    TOTAL_DEBE = TOTAL_DEBE + MOV_DEBE;
    TOTAL_HABER = TOTAL_HABER + MOV_HABER;

    update MOVIMIENTOS set
           SALDO = :TOTAL_SALDO
    where ID = :MOV_ID;
  end
end
^

SET TERM ; ^

Al ejecutarse todo el procedimiento almacenado en el servidor, te ahorras todo el movimiento de red necesario para recorrer los 2.000 registros, recuperar los valores de sus campos, modificar el registro, etc. ...

El paso por la red suele ser el cuello de botella de estos procesos. Mediante la ejecución en un procedimiento almacenado, por la red solo se pasa la solicitud de ejecución del procedimiento almacenado, y se reciben como resultado los valores de Saldo, Debe y Haber Totales.

Saludos.
__________________
Marc Guillot (Hi ha 10 tipus de persones, els que saben binari i els que no).
Responder Con Cita
  #3  
Antiguo 08-09-2011
Bretema Bretema is offline
Miembro
 
Registrado: nov 2003
Ubicación: Vigo - Galicia
Posts: 37
Poder: 0
Bretema Va por buen camino
Muchas gracias por tu respuesta Guillotmarc, he puesto en marcha la solución que me propusiste y efectivamente el descenso en el tiempo de proceso ha sido apreciable, el tiempo de proceso de los registros se ha reducido a un tercio del anterior.

Saludos cordiales.
Responder Con Cita
  #4  
Antiguo 09-09-2011
Avatar de guillotmarc
guillotmarc guillotmarc is offline
Miembro
 
Registrado: may 2003
Ubicación: Huelva
Posts: 2.638
Poder: 24
guillotmarc Va por buen camino
Cita:
Empezado por Bretema Ver Mensaje
Muchas gracias por tu respuesta Guillotmarc, he puesto en marcha la solución que me propusiste y efectivamente el descenso en el tiempo de proceso ha sido apreciable, el tiempo de proceso de los registros se ha reducido a un tercio del anterior.

Saludos cordiales.
Sinceramente, 10 segundos (un tercio de los 30 segundos originales), me siguen pareciendo excesivos para solo 2.000 registros.

Parece que la base de datos tiene algún trigger para AFTER UPDATE en la tabla MOVIMIENTOS, y que la ejecución de ese trigger para cada registro ralentiza todo el proceso.

Deberías considerar desactivar los triggers que no sean imprescindibles, antes de lanzar el proceso. Una vez finalizado el proceso, ya puedes reactivar de nuevo los triggers.

ALTER TRIGGER MOVIMIENTOS_AU01 INACTIVE;
ALTER TRIGGER MOVIMIENTOS_AU02 INACTIVE;
EXECUTE PROCEDURE ACTUALIZAR_SALDO;
ALTER TRIGGER MOVIMIENTOS_AU01 ACTIVE;
ALTER TRIGGER MOVIMIENTOS_AU02 ACTIVE;

Saludos.
__________________
Marc Guillot (Hi ha 10 tipus de persones, els que saben binari i els que no).
Responder Con Cita
  #5  
Antiguo 09-09-2011
Avatar de Casimiro Notevi
Casimiro Notevi Casimiro Notevi is offline
Moderador
 
Registrado: sep 2004
Ubicación: En algún lugar.
Posts: 32.057
Poder: 10
Casimiro Notevi Tiene un aura espectacularCasimiro Notevi Tiene un aura espectacular
Cita:
Empezado por guillotmarc Ver Mensaje
Sinceramente, 10 segundos (un tercio de los 30 segundos originales), me siguen pareciendo excesivos para solo 2.000 registros.
Yo uso un store procedure similar y con bastantes más de 2000 registros tarda apenas una décima de segundo. Así que es seguro que sea lo que comenta guillotmarc, algún trigger.
Responder Con Cita
  #6  
Antiguo 09-09-2011
Bretema Bretema is offline
Miembro
 
Registrado: nov 2003
Ubicación: Vigo - Galicia
Posts: 37
Poder: 0
Bretema Va por buen camino
He revisado la tabla en cuestión y no tiene triggers, de hecho es una tabla que solo se usa para volcar los registros desde un par de tablas y después realizar el proceso que comenté para establecer el saldo tras cada movimiento. Una vez el usuario lo ve, cuando cierra la ventana se vacia la tabla. El tiempo lo mido exclusivamente desde justo la llamada a la store proc. hasta que finaliza ..... en mi dual core tardaba unos siete segundos y bajó a dos ... en otro mas lento, un amd con xp y un mega de ram en el cual tardaba 17 segundos bajó a unos seis. El ahorro es evidente pero desde luego lejos de las décimas que comenta Casimiro .... no se ya si será cuestión de tocar algún parámetro del gestor de la base de datos ... la página de la tabla la tengo definida en 4096 que creo es el estandar.

Saludos.
Responder Con Cita
  #7  
Antiguo 09-09-2011
Avatar de Casimiro Notevi
Casimiro Notevi Casimiro Notevi is offline
Moderador
 
Registrado: sep 2004
Ubicación: En algún lugar.
Posts: 32.057
Poder: 10
Casimiro Notevi Tiene un aura espectacularCasimiro Notevi Tiene un aura espectacular
Supongo que no será posible probar la base de datos, ¿verdad?.

Por cierto, haz un backup y luego restauras a tamaño de página 8192, debe de ir bastante mejor.
Responder Con Cita
  #8  
Antiguo 10-09-2011
Avatar de newtron
[newtron] newtron is offline
Membrillo Premium
 
Registrado: abr 2007
Ubicación: Motril, Granada
Posts: 3.474
Poder: 21
newtron Va camino a la fama
Igual suelto una tontería porque yo no uso firebird pero ¿no sería posible crear esa tabla en memoria?, eso agilizaría bastante.

Saludos
__________________
Be water my friend.
Responder Con Cita
  #9  
Antiguo 10-09-2011
Avatar de Casimiro Notevi
Casimiro Notevi Casimiro Notevi is offline
Moderador
 
Registrado: sep 2004
Ubicación: En algún lugar.
Posts: 32.057
Poder: 10
Casimiro Notevi Tiene un aura espectacularCasimiro Notevi Tiene un aura espectacular
Es que realmente debe tener algún problema añadido, 2.000 registros es nada, es como si dice que son 2 registros, igual, debe ser algo instantáneo.
Responder Con Cita
  #10  
Antiguo 12-09-2011
JXJ JXJ is offline
Miembro
 
Registrado: abr 2005
Posts: 2.475
Poder: 22
JXJ Va por buen camino
Cita:
Empezado por guillotmarc Ver Mensaje
Esto es un ejemplo perfecto de un proceso que se ejecuta muchísimo más rápido en un procedimiento almacenado.

Sería más o menos así :

Código SQL [-]SET TERM ^ ; create procedure ACTUALIZAR_SALDO returns ( TOTAL_SALDO numeric(15,2), TOTAL_DEBE numeric(15,2), TOTAL_HABER numeric(15,2)) as declare variable MOV_ID integer; declare variable MOV_DEBE numeric(15,2); declare variable MOV_HABER numeric(15,2); begin TOTAL_SALDO = 0; TOTAL_DEBE = 0; TOTAL_HABER = 0; for select ID, coalesce(DEBE,0), coalesce(HABER,0) from MOVIMIENTOS into :MOV_ID, :MOV_DEBE, :MOV_HABER do begin TOTAL_SALDO = TOTAL_SALDO + MOV_DEBE - MOV_HABER; TOTAL_DEBE = TOTAL_DEBE + MOV_DEBE; TOTAL_HABER = TOTAL_HABER + MOV_HABER; update MOVIMIENTOS set SALDO = :TOTAL_SALDO where ID = :MOV_ID; end end ^ SET TERM ; ^


Al ejecutarse todo el procedimiento almacenado en el servidor, te ahorras todo el movimiento de red necesario para recorrer los 2.000 registros, recuperar los valores de sus campos, modificar el registro, etc. ...

El paso por la red suele ser el cuello de botella de estos procesos. Mediante la ejecución en un procedimiento almacenado, por la red solo se pasa la solicitud de ejecución del procedimiento almacenado, y se reciben como resultado los valores de Saldo, Debe y Haber Totales.

Saludos.
hola

http://www.clubdelphi.com/foros/imag...ine=1267189133 guillotmarc
Registrado

tu codigo me parece muy interesante.

com puedo aprender mas.

dijamos si quiero hacer unos reportes de ventas
y quiero mostrar la informacion ordenada
por ventas por cliente. por dia. por tipo de pago.

apenas se me ocurrio que esto de los stored procedures
me serviria.
Responder Con Cita
  #11  
Antiguo 20-09-2011
Bretema Bretema is offline
Miembro
 
Registrado: nov 2003
Ubicación: Vigo - Galicia
Posts: 37
Poder: 0
Bretema Va por buen camino
Thumbs up

Al final solucioné el problema atacándolo desde dos frentes, el proceso de cálculo del saldo lo hago con una store procedure tal y como me sugirió guillotmarc, con esto tengo una reducción apreciable del tiempo de proceso. La tabla en cuestión arrastra movimientos de varios años (exigencias del usuario), pero para cortar el problema de raíz puse en el formulario un campo para dar la opción de seleccionar un año en concreto o manejar todo el conjunto. Hago la select sobre ese campo y con ello me aseguro de que siempre habrá una opción para que el proceso sea rápido aunque la tabla se vaya cargando de nuevo de registros .... como el saldo inicial de la cuenta no es anual, tuve que crear otra store procedure para calcular el saldo inicial en función del año .... al final la consulta de un año concreto es casi instantáneo.

De todas formas me queda el mosqueo de que incluso con la primera solución de guillomarc siguiese tardando tanto, sin triggers y sobre una tabla temporal exclusivamente para ese proceso ..... también probé la sugerencia de Casimiro pero el variar el tamaño de página en este caso no tuvo demasiada incidencia.

Gracias a todos por vuestras sugerencias.
Responder Con Cita
  #12  
Antiguo 21-09-2011
Avatar de Delphius
[Delphius] Delphius is offline
Miembro Premium
 
Registrado: jul 2004
Ubicación: Salta, Argentina
Posts: 5.582
Poder: 25
Delphius Va camino a la fama
Hay algo que no se dijo... ¿Y, cómo andamos con los índices?

Saludos,
__________________
Delphius
[Guia de estilo][Buscar]
Responder Con Cita
  #13  
Antiguo 21-09-2011
Avatar de guillotmarc
guillotmarc guillotmarc is offline
Miembro
 
Registrado: may 2003
Ubicación: Huelva
Posts: 2.638
Poder: 24
guillotmarc Va por buen camino
Cita:
Empezado por Delphius Ver Mensaje
Hay algo que no se dijo... ¿Y, cómo andamos con los índices?

Saludos,
Muchísimos índices tendría que tener para provocar esos retardos. También lo pensé, pero no creo que sea la causa.

A mi en todo momento me ha parecido que por algún lado se está disparando código adicional (mediante eventos en Delphi o triggers en Firebird), pero no lo ha encontrado. Así que no sé que pensar.
__________________
Marc Guillot (Hi ha 10 tipus de persones, els que saben binari i els que no).
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
Lentitud en ADO JF Sebastian Conexión con bases de datos 0 16-03-2008 11:06:26
Lentitud en la red. Carlos Arevalo Varios 2 09-07-2007 19:04:08
Lentitud en la aplicacion cesarjbf Gráficos 2 24-10-2003 02:46:38
Lentitud en la red Ignacio Conexión con bases de datos 2 26-08-2003 14:09:39


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


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