PDA

Ver la Versión Completa : ¿liberar memoria despues de usar una Query?


mrmanuel
12-10-2005, 11:34:35
Hola a todos.

Mi problema es la siguiente:

Estoy haciendo un listado sobre tabla DBase, usando sentencia SQL sobre una TQuery, cuando lleva evaluados 400 registros, me da un error de Memoria Insuficiente.
He medio resuelto el problema, poniendo un rango de busqueda de 300 registros cada vez, es decir del 1-300, del 300 al 600, etc.

El problema esta q los primeros 300 lo hace perfectamente, pero cuando le digo que liste el siguiente rango de 300, me vuelve a cascar con el problema de la Memoria. (Sigue siendo al evaluar el que hace 400).

He probado a poner:
Query.Free
Query.Sql.Free
despues de cada listado y en ambos casos, al intentar listar el segundo rango, que salta una excepción de Delphi.


¿Como podría liberar la memoria de una sentencia sql (Query), para cuando vuelva a usar dicha query este 100% libre?

Uso Delphi 5.


Gracias

Lepe
12-10-2005, 12:11:29
Pon el código para crear tu TQuery.

Si la creas en tiempo de diseño, solo tienes que cerrarla con query1.Close, modificar el query.sql.text y despues volverla a abrir.

saludos

lucasarts_18
12-10-2005, 15:07:39
Si la creas en tiempo de diseño, solo tienes que cerrarla con query1.Close
saludos
Una duda sobre esto, es decir el mismo formulario o contenedor se encarga de destruir el objeto ??? :confused:. Yo sospecho que así es..

:cool:

lucasarts_18
12-10-2005, 15:28:44
Hola:

Gracias a un Post de Roman, he leído que un Form al momento de destruirse llama automaticamente a un método DestroyComponents, esta es la respuesta..:D

Delfino
12-10-2005, 15:38:14
Aun asi conviene a lo mejor destruirla antes de cerrar el formulario..

Lepe
12-10-2005, 17:57:03
Aun asi conviene a lo mejor destruirla antes de cerrar el formulario.


Me ha gustado esta respuesta. No deja claro si hay que destruirla o no, pero tampoco deja claro que no deba hacerse :D.

La respuesta es: No hay que destruirla, es más ni siquiera cerrarla (query.Close), ya que al destruirse el contenedor, llama al destructor de la consulta, y si está abierta, primero se cierra y despues se destruye.

Gydba
12-10-2005, 18:38:16
Buenas gentes,

No se si viene de la mano del comentario de Delfino pero en mi caso siempre libero los objetos apenas dejo de utilizarlos así no siguen dando vueltas a lo largo de la vida útil de su owner.

Chauchas!!

ContraVeneno
12-10-2005, 19:19:47
espero que mrmanuel coloque su código para poder saber si le propongo que haga: TuQuery.SQL.Clear antes de usarla.

Delfino
13-10-2005, 14:00:17
No hay que destruirla, es más ni siquiera cerrarla (query.Close)

Siempre pongo dataset.Close antes de cerrarse el form, eso pq he tenido varios problemas si no se llama al metodo close sobre todo cuando hay codigo en el evento datachange del datasource asociado, no se si ha pasado a otros.

Ademas segun este (http://delphi.about.com/od/kbcurt/ss/dynamiccreateno.htm) articulo es conveniente crear componentes con el owner nil y destruirlos..

mrmanuel
21-10-2005, 19:18:44
Hola a todos.

siento el retraso, pero problemas con el curro.

Bueno os comento. la Query la creo en tiempo de diseño, la cree simplemente añadiendola en una form, donde tengo una serie de tablas y de queris auxiliares.

Dicho form está activo en todo el momento, ya que desde el hago todas las operaciones de acceso a los datos. He probado a poner Query.Close, y nada.

respecto a lo de cerrar el form y que el se encarge de destruir los objetos, no lo puedo hacer ya que como he dicho antes, es el que lleva toda la chicha del programa.

he provado a liberar el la query en concreto, pero al intentar acceder a ella nuevamente, me dice que no, ya que le he liberado anteriormente y da problemas de acceso a memoria.

Lo que tendria que hacer es que liberarse la memoria usada para esa consulta, pero que no me cascase al intentar acceder a ella.

Gracias.

ContraVeneno
21-10-2005, 19:39:33
with TuQuery do begin
if Active then Close;
SQL.Clear;
SQL.Add('Select... etc');
SQL.Add('From ... etc');
SQL.Add('WHere.... etc');
Open;
end; //with


a ver si eso te sirve...
al menos es lo que a mi me funciona y puedo cambiar el Query cuantas veces sea necesario....

mrmanuel
10-11-2005, 13:04:26
Hola nuevamente.

el codigo que me indica CntraVeneno, es tal y como yo lo hago, lo único que despues del OPEN, hago un Execute.

yo lo que quiero hacer es liberar la memoria q usa esa query en un momento dado, sobre tado cuando hago consultas con gran nº de registros.

el problema que tengo es q se me queda en memoria, creo, y cuando intento hacer cualquier tipo de consulta, ya sea en la misma query o en otra, me da error de acceso a memoria.

gracias

lucasarts_18
10-11-2005, 13:41:30
el codigo que me indica CntraVeneno, es tal y como yo lo hago, lo único que despues del OPEN, hago un Execute.

Execute por qué ?, bastaría con un Open si la consulta es un Select


yo lo que quiero hacer es liberar la memoria q usa esa query en un momento dado, sobre tado cuando hago consultas con gran nº de registros.
el problema que tengo es q se me queda en memoria, creo, y cuando intento hacer cualquier tipo de consulta, ya sea en la misma query o en otra, me da error de acceso a memoria.

Si lo liberas y luegos deseas usarlo nuevamente, debes crearlo nuevamente..

Hasta Luego -

ContraVeneno
10-11-2005, 16:38:39
Execute por qué ? :confused: si, ¿por qué?
Open cuando es consulta (Select)
Execute cuando es modificación (Update, Delete, Insert, alter, etc.)

¿nos podrías decir exactamente cuando haces que cosa te marca el error? y ¿cuál es el error específico que te marca?

Lepe
10-11-2005, 17:36:05
Si despues de 12 mensajes, no ha mostrado el código, será porque pertenece a MICROSOFT, fijo :p :p .

saludos

lucasarts_18
10-11-2005, 17:42:04
el problema que tengo es q se me queda en memoria, creo
Asegurate con FreandNil(Objeto)

y cuando intento hacer cualquier tipo de consulta, ya sea en la misma query
Aquí usas el mismo query, que al parecer ya has liberado y por lo tanto te da una excepción.

o en otra, me da error de acceso a memoria.
Este seguramente no lo haz creado.


¿cuál es el error específico que te marca?
Es un error de los típico de numeros hexadecimales y cosas raras

Si despues de 12 mensajes, no ha mostrado el código, será porque pertenece a MICROSOFT
:D:D:D

Hasta Luego -

Lepe
10-11-2005, 18:13:02
Ademas segun este (http://delphi.about.com/od/kbcurt/ss/dynamiccreateno.htm) articulo es conveniente crear componentes con el owner nil y destruirlos..

A ver, a ver que son cosas distintas.... no podemos mezclar cosas, y si lo hacemos, debemos tener cuidado.

Si le ponemos a un objeto el Owner como Form1, Application etc, jamás llamaremos a Query1.Free en nuestro código, repito, jamás. ¿Por qué?, entendamos que significa Query1, que es lo mismo que cualquier otro objeto, ventana, clase de delphi, etc.

Query1 es una variable de tipo puntero. Si la ponemos en modo diseño, Delphi se encarga de crearla en memoria cuando se ejecute nuestro programa, y además el Owner de ese Query será el Form que lo contenga. Por tanto, cuando se libere esa ventana, se liberará el Query1.

Si en esa misma ventana colocamos un Query1.Free, la consulta se libera de memoria RAM, pero la variable Query1 se queda apuntando a la zona de memoria donde estaba creada. La próxima vez que se acceda a Query1 nos dará un access Violation. Es más, como ha sido puesta en modo diseño, al liberar la ventana, ésta intentará liberar la consulta, como Query1 es un puntero que tiene un valor, irá a esa zona para liberarla, y entonces obtendremos otro Access Violation.

Estamos en la situación de que un objeto tiene un Owner y queremos liberarlo de la memoria antes que su Owner, ¿no hay una solución? Si, usamos FreeAndNil(Query1); nosotros ponemos FreeAndnil(Query1), se libera la consulta de la RAM y además esa rutina hace Query1 := nil , y aquí está el truco; cuando se vaya a liberar el Form, éste pregunta ¿Query1 <> nil?, como es igual a nil, el Form entiende que ya ha sido liberado anteriormente y no trata de liberarlo de nuevo. No tendrá fallos nuestro código.

Otra situacións: Nosotros creamos un objeto y le decimos

variable := Tvariable.Create(nil);

es decir, no tiene Owner, por tanto, será mejor que nosotros la destruyamos. Para evitar errores en una segunda o tercera creación, tambien usaremos FreeAndNil(variable), si es nil, no ha sido creada previamente, o bien ha sido liberada. Si es distinto de nil, es que está creada, no debemos crearla de nuevo.

Si usamos variable.Free , variable se queda apuntando a una zona inválida de la memoria, por tanto, no sabemos si está creada, o ya fue liberada. Al usar FreeAndNil, no tenemos esa ambigüedad:


Tform1 = class
private
query1:Tquery;
end;

procedure TForm1.Button1click(...);
begin
if query1 = nil then
Query1 := Tquery.Create(nil)
else
Showmessage('ya está creada')
end;

procedure TForm1.Button2click(...);
begin
FreeandNil(query1);
end;

procedure TForm1.Button3click(...);
begin
query1.Free
end;


Probemos a dar al boton1 y despues 2 o 3 veces al boton2, no ocurre nada ¿verdad?, el programa funciona bien.

Probemos a dar al boton1 y despues 2 veces al boton3.

Existe una situación algo más rara, es crear un componente en tiempo de ejecución y no liberarlo.

Query1 := Tquery.Create(form1);

Y nos olvidamos de FreeAndNil y de .Free, porque al fin y al cabo, le hemos puesto el dueño, él se encargará de destruirlo. Que es lo que hacemos con los TEdits, Grids, Buttons, que ponemos en nuestras ventanas en tiempo de diseño.

saludos y espero se entienda.

lucasarts_18
10-11-2005, 18:38:35
Qué explicación lepe !!!!!!!
Con esto debería quedar más que claro este hilo :D

Hasta Luego -

Lepe
10-11-2005, 18:45:44
ups, pues lee de nuevo que lo he editado :D

saludos

vtdeleon
10-11-2005, 19:54:46
Saludos

No se si esto ya se ha dicho, he leido tanto este hilo que ya no sé de que se trata:p


mrmanuel
Muestranos el codigo que utilizas para:
Crear el Query,
Transaccion(un poquito)
y para liberarla.

Puede ser otra cosa que esté consumiendo :o

Lepe
11-11-2005, 01:07:39
vtdeleon, las explicaciones no salen siempre como uno quiere ;).


Pon el código para crear tu TQuery.

Eso en la primera respuesta del hilo :D

saludos

mrmanuel
01-12-2005, 15:07:18
hola a todos.

siento el retraso y agradezco a todos los compañeros que han respondido a este hilo. no he podido postear el codigo porque he estado muy liado en el trabajo.

Pero prometo que esta tarde-noche postie dicho código.

Disculpen la tardanza.

Gracias

kcbb
31-10-2006, 20:58:18
Buenas tengo el siguiente problema, ejecuto un Query, pero se queda en memoria, he usado varias formas para tratar de liberarlo y nada, estas son las que he usado
QRY4.EnableControls;
QRY4.DisableControls;
QRY4.First;
FreeAndNil(qry4);
QRY4.free;
QRY4.Destroy;
QRY4.DestroyComponents;
QRY4.FreeOnRelease;
QRY4.RemoveComponent(QRY4);
QRY4.CleanupInstance;

Lepe
01-11-2006, 01:47:51
Dependiendo de cómo se crea y donde se crea esa consulta, habrá que usar una forma u otra de liberarlo.

Por regla general, si Delphi lo crea, que Delphi lo destruya.

Me conviene destruirlo a mi, entonces lo creo yo y lo libero yo.

Hay otras variantes, pero yo al menos uso esas 2.

¿como sabes que se queda en memoria?

saludos

kcbb
01-11-2006, 03:55:26
por que cierro el programa y se queda el ejecutable en el administrador de tareas, esto queda en background, tambien te das cuenta pq delphi que corriendo peros los formularios cerrados. Pero esto sucede nada mas cuando uso el query. Otra cosa que me he dado cuenta es que nada mas pasa con Base de Datos en Pervasic ( que es una de las que uso en la empresa, la otra es AS400 y no da este error).

Lepe
01-11-2006, 14:20:14
En principio, lo correcto sería:

qry4.Close;
FreeAndnil(qry4);


Aunque me gustaría ver como creas el query.

Si lo haces con qry4 := Tquery.Create(nil) (ojo al parámetro nil) o bien la creas en tiempo de diseño, no deberías tener problemas.

Puede que no sea por la destrucción de ese query, sino por efectos colaterales que ésta produzca. Para que te hagas una idea de lo que quiero decir: puede que en el AfterClose de esa query, intentes abrirla de nuevo y por tanto jamás se cerrará por completo, o incluso estás abriendo otra consulta, form o similar en ese evento o uno parecido.

Lamento no tener una respuesta directa. Espero que te dé algunos indicios para conseguir solucionar el problema.

Saludos