Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   OOP (https://www.clubdelphi.com/foros/forumdisplay.php?f=5)
-   -   FreeAndNil VS Free (https://www.clubdelphi.com/foros/showthread.php?t=73190)

GerTorresM 06-04-2011 17:27:04

FreeAndNil VS Free
 
Hola a tod@s:

Bien la duda que me asalta el día de hoy es la siguiente:

cual es la función mas indicada para destruir un objeto para este caso un formulario

Código Delphi [-]
procedure TFMenuPrincipal.ANovedadesExecute(Sender: TObject);
Var FNovedades_temporal : TFNovedades;
begin
  try
    FNovedades_temporal:= TFNovedades.Create(self);
    FNovedades_temporal.ShowModal;
  finally
    FNovedades_temporal.Free;
  end;
end;

o

Código Delphi [-]
procedure TFMenuPrincipal.ANovedadesExecute(Sender: TObject);
Var FNovedades_temporal : TFNovedades;
begin
  try
    FNovedades_temporal:= TFNovedades.Create(self);
    FNovedades_temporal.ShowModal;
  finally
    FreeAndNil(FNovedades_temporal);
  end;
end;
y muy en concreto cual es la diferencia entre las dos funciones


Agradezco de antemano



gertorresm
Colombia

oscarac 06-04-2011 17:32:17

muy buena pregunta....
no me habia puesto a indagar sobre el tema

yo libero los formularios asi

Código Delphi [-]
 
frmBalance := nil;

roman 06-04-2011 17:42:24

Cita:

Empezado por GerTorresM (Mensaje 396044)
cual es la función mas indicada para destruir un objeto para este caso un formulario

[...]

y muy en concreto cual es la diferencia entre las dos funciones

En términos llanos, FreeAndNil lo que hace es llamar al método Free del objeto y luego asignar nil a la variable que hace referencia al objeto. Dicho de otra forma:

Código Delphi [-]
FreeAndNil(Obj);

es equivalente a

Código Delphi [-]
Obj.Free;
Obj := nil;

La única razón para usar FreeAndNil al momento de liberar un objeto es asegurarte que el objeto apunta a nil y no a una dirección de memoria arbitraria. Esto te puede servir si posteriormente utilizas la variable y quieres saber si sigue "vivo":

Código Delphi [-]
if Assigned(Obj) then
  ...

Pero, en el caso que propones, siendo una variable temporal que no se usa posteriormente, no tiene nonguna ventaja poner la variable en nil ni, por tanto, usar FreeAndNil.

// Saludos

roman 06-04-2011 17:43:19

Cita:

Empezado por oscarac (Mensaje 396045)
yo libero los formularios asi

Código Delphi [-]
 
frmBalance := nil;

Así no estás liberado nada. Simplemente apuntas la variable frmBalance a nil pero el objeo sigue existiendo en memoria. Tienes que usar Free.

// Saludos

gluglu 06-04-2011 17:43:54

Para preguntar en cualquier parte de tu programa si uno de tus formularios concretos está o no creado, al menos yo, pregunto por
Código Delphi [-]
if Asigned(MiForm) then ....
La única manera de que te devuelva False, es decir, que el Formulario no está creado, es que MiForm sea igual a Nil

Si Haces sólo un Free del formulario, no se pone a 'Nil' y por lo tanto la pregunta anterior siempre te dará True incluso aunque hayas 'liberado' tu formulario con Free.

Por eso FreeAndNil, o simplemente también sirve :
Código Delphi [-]
MiForm.Free
MiForm := nil;

... sobre gustos ...

gluglu 06-04-2011 17:44:46

... se me adelantó el Maestro Roman !! :o

oscarac 06-04-2011 17:45:14

como comentario adicional encontre esto que quiza te ayude

oscarac 06-04-2011 18:05:55

mmm me puse a cambiar los...

frmBalance := nil

por

FreeandNil(frmBalance)

y me sale el siguiente mensaje de error cuando cierro un formulario

Invalid Pointer operation


what da faq????

oscarac 06-04-2011 18:21:37

la forma como creo los formularios es asi (dentro de un formulario Main)

Código Delphi [-]
 
frmBalance := TfrmBalance.create(nil);
frmbalance.show;


cuando coloco

freeandnil(frmbalance) en el destroy o en el close....me sale el error mencionado

pero si coloco esto en el destroy no me aparece mensaje

Código Delphi [-]
 
frmbalance := nil;
frmBalance.free;

me libera ? si? no?

antes de llamar al formulario balance coloque esto

Código Delphi [-]
 
if frmBalance = nil
  frmBalance := TfrmBalance.create(nil);
y siempre es nil

me olvidaba
coloco

action := caFree; en el close

esta bien asi?

roman 06-04-2011 18:23:28

Habría que ver el contexto. Por ejemplo, si llamas FreeAndNil desde el mismo formulario frmBalance, podría suceder el error que pones. En ese caso tendrías que usar Release en lugar de Free.

// Saludos

oscarac 06-04-2011 18:26:53

me atrevo a preguntar

cual es la diferencia entre

frmBalance := TfrmBalance.create(nil) ;

y

frmBalance := TfrmBalance.create(self);

oscarac 06-04-2011 18:28:54

Cita:

Empezado por roman (Mensaje 396054)
Habría que ver el contexto. Por ejemplo, si llamas FreeAndNil desde el mismo formulario frmBalance, podría suceder el error que pones. En ese caso tendrías que usar Release en lugar de Free.

// Saludos

asi es.. lo estaba llamando dentro del mismo formulario frmBalance

entonces lo correcto seria

frmBalance.nil;
frmBalance.free;

???

Casimiro Notevi 06-04-2011 18:29:42

No, no puedes hacerlo así, primero hay que liberar la memoria y luego asignarle nil a la misma.
Por eso el orden correcto es

x.free;
x:=nil;

o más cómodo:

freeandnil(x);

Si lo haces al revés, asignas nil, entonces cuando llamas a free no hace nada porque está a nil.


Edito: he contestado lo de antes sin haber visto tu último mensaje :)

oscarac 06-04-2011 18:33:32

Concluyendo....

primero free y despues nil

y nunca nu freeandnil dentro del mismo form

Casimiro Notevi 06-04-2011 18:45:05

Cita:

Empezado por oscarac (Mensaje 396059)
Concluyendo....
primero free y despues nil
y nunca nu freeandnil dentro del mismo form

Eso sería como levantarte tú mismo del suelo tirando de los cordones de tus botas :D

El boot :)

oscarac 06-04-2011 18:48:09

mmmm
deberias ver a los acrobatas que hay por aca...
desafian las reglas de la fisica :D:D:D

maeyanes 06-04-2011 19:05:04

Hola...

oscarac, no es recomendable usar la variable que declara Delphi cuando creas un formulario nuevo:

Código Delphi [-]
type
  TMyForm = class(TForm)
  // ...
  end;

var
  MyForm: TMyForm; // <--- esta

dentro de los métodos o manejadores de eventos del mismo.

Estó es, si haces algo como:

Código Delphi [-]
procedure TMyForm.FormDestroy(Sender: TObject);
begin
  MyForm := nil
end;

Te podría traer problemas posteriores.

Si quieres garantizar que un formulario se destruye al cerrarlo una de las formas de lograrlo es el evento OnClose:

Código Delphi [-]
procedure TMyForm.FormClose(Sender: TObject; var Action: TCloseAction);
begin
  Action := caFree
end;


Saludos...

oscarac 06-04-2011 19:15:44

sigo con el tema...

coloco

frmbalance.free
frmbalance.nil

y me sale stack overflow :confused::confused::confused:

rgstuamigo 06-04-2011 19:25:34

Cita:

Empezado por oscarac (Mensaje 396066)
sigo con el tema...

coloco

frmbalance.free
frmbalance.nil

y me sale stack overflow :confused::confused::confused:

Pero.. en qué parte de tu código fuente colocas esas instrucciones? ¿Algun evento específico? ¿dónde?:confused:
Saludos...:)

maeyanes 06-04-2011 19:28:09

Hola...

De seguro está poniendo ese código en el evento OnDestroy del formulario en cuestión...


Saludos...


La franja horaria es GMT +2. Ahora son las 21:41:07.

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