![]() |
Warum macht ein TForm.Close nicht NIL
Hi.
Ein TForm.Close lässt eigentlich vermuten, dass das Objekt weg ist. Aber das ist es nicht immer, bzw. nicht sofort.
Delphi-Quellcode:
Damit ich auf NIL prüfen kann musste ich noch das 'Close' ergänzen
MyForm := TMyForm.Create(MainForm);
MyForm.Show; ... ... MyForm.Close; ... ... if MyForm <> NIL then begin ShowMessage('Bin verwirrt. Warum bist du nicht NIL???') end;
Delphi-Quellcode:
Meine Frage: Ist das nun gut oder schlecht. Wenn ein Objekt auf NIL gesetzt wird was ist dann mit dem Speicher. Räumt Delphi trotzdem auf oder hab ich ein Speicherleck??
MyForm.Close;
FreeAndNil(MyForm); Ein 'TForm.Close' macht einen 'Release'. Bisher dachte ich damit wäre alles aus dem Speicher raus. Jetzt musste ich mit Erstaunen feststellen, dass dem nicht so ist. |
Re: Herr 'Close' und Frau 'NIL'
So amüsant wie dein Threadtitel auch ist, er ist nicht sehr hilfreich, wenn es um die kurze Beschreibung deines Problems geht., Bitte ändere ihn in einen aussagekräftigen Titel.
|
Re: Herr 'Close' und Frau 'NIL'
Ein Glück, dass es nicht so ist.
Die Methode "Close" schließt das Fenster nur, aber es bleibt weiterhin vollständig im Speicher vorhanden und wartet nur darauf, mittels "Show()" wieder angezeigt zu werden. Bei Fenstern gelten leicht andere Spielregeln, hier solltest Du im Allgemeinen die Methode "Release" nehmen, um es aus dem Speicher zu entfernen. Das erlaubt dem Fenster, noch ein paar Aufräumarbeiten zu erledigen. [edit] ... und dann natürlich das, was Luckie in Bezug auf den Thementitel sagte. [/edit] |
Re: Herr 'Close' und Frau 'NIL'
Hi,
das Objekt wird bei Close nicht unbedingt freigegeben, das hängt davon ab, was du im OnClose zurückgibst (Action = caFree --> Freigabe, ansonsten nicht). Mit .Free oder FreeAndNil hingegen wird das Formularobjekt immer freigegeben, aber nur bei FreeAndNil wird es auch auf nil gesetzt, ansonsten verweist deine Variable immer noch an die Stelle, an der das Objekt vorher lag. [edit] Wie wärs denn damit:
Delphi-Quellcode:
[/edit]
procedure ReleaseAndNil(var X: TCustomForm); //inline;
begin if Assigned(X) then begin X.Release; X := nil; end; end; Mfg FAlter |
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
Auszug Forms.pas
Delphi-Quellcode:
procedure TCustomForm.Close;
var CloseAction: TCloseAction; begin if fsModal in FFormState then ModalResult := mrCancel else if CloseQuery then begin if FormStyle = fsMDIChild then if biMinimize in BorderIcons then CloseAction := caMinimize else CloseAction := caNone else CloseAction := caHide; DoClose(CloseAction); if CloseAction <> caNone then if Application.MainForm = Self then Application.Terminate else if CloseAction = caHide then Hide else if CloseAction = caMinimize then WindowState := wsMinimized else Release; end; end; In der Delpi Hilfe steh: Mit Release können Sie das Formular aus dem Speicher entfernen. Dieser Info habe ich auch bisher Glauben gescheckt. |
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
|
Re: Warum macht ein TForm.Close nicht NIL
![]() Zitat:
|
Re: Warum macht ein TForm.Close nicht NIL
Ohhh neee :wall:
Jetzt hab ichs gesehen. Wenn ich im OnClose keine CloseAction definiere (z.B. caFree), macht es ein caHide. Aber wie nun:
Delphi-Quellcode:
oder
MyForm.Close;
MyForm.Release;
Delphi-Quellcode:
MyForm.Close;
FreeAndNil(MYform) |
Re: Warum macht ein TForm.Close nicht NIL
Nur MyForm.Release für deine Zwecke.
|
Re: Warum macht ein TForm.Close nicht NIL
Wird die Variable noch genutzt dann FreeAndNil() sonst MyForm.Free;
|
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
Deine Variable ist nur eine Referenz auf den Speicher des Formulars. Du bist wirklich der Meinung, dass Delphi einen riesigen Verwaltungsaufwand treibt um im folgenden Beispiel alle diese Variablen auf NIL zu setzen, weil sie alle auf das eine Formular verweist?
Delphi-Quellcode:
Anderes Beispiel:
var
Form1, Form2, Form3, Form4, Form5, FormWhatever: TForm; begin Form1 := TForm.Create(nil); // eine Instanz Form2 := Form1; Form3 := Form1; Form4 := Form1; Form5 := Form1; FormWhatever := Form1; // Nun zeigen alle diese Variablen auf dieses eine Formular. Form1.Release; // Formular freigeben // Nun ist der vom Formular belegte Speicher freigegeben. Aber alle Variablen von Form1 bis FormWhatever // enthalten aber immernoch die Adresse die das Formular hatte. Nur der Speicher an der Adresse ist wieder // verfügbar. // Irgendwelche Zugriffe auf Form1 bis FormWhatever sind tödlich! // Delphi müsste einen riesigen Verwaltungsaufwand treiben um alle Variablen auf NIL zu setzen. Schon allein // folgendes setzt (natürlich) nicht gleich alle Variablen auf nil: Form1 := nil; // Hiernach sind Form2 bis FormWhatever noch immer mit der alten Adresse gefüllt. Ein Freund von dir zieht um und hat eine neue Adresse. Du schreibst seine neue Adresse in dein Adressbuch, aber trotzdem sind nicht automatisch alle anderen Stellen wo die Adresse vermerkt wurde mit geändert: Outlook Buch, Telefon, etc. Die Variablen die die Instanz der Form aufnehmen sind an unterschiedlicher Stelle im Speicher und sind jeder für sich. Diese Variablen verweisen nur an eine Stelle mit der Form Instanz. |
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
|
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
Ich hab bisher immer caFree oder eben FreeAndNil verwendet. Wenn ich nun Release statt FreeAndNil verwende, muss ich doch MyForm nach dem Release noch "nilen", oder? Wenn ja, gibt das dann keine unschönen Nebeneffekte, da Release laut Hilfe nicht wartet bis alles freigegeben wurde und so man theoretisch MyForm schon Nil zuweist bevor Release im Hintergrund überhaupt fertig ist. Ist die CloseAction caFree eher mit FreeAndNil oder mit Release vergleichbar? |
Re: Warum macht ein TForm.Close nicht NIL
Daher habe ich ja geschrieben, wenn genutzt wird FreeAndNil()
|
Re: Warum macht ein TForm.Close nicht NIL
Oh, da hab ich doch glatt was vertauscht - sorry.
|
Re: Warum macht ein TForm.Close nicht NIL
@hitzi
caFree ist dann wohl eher ein Release und kein FreeAndNil. Wenn also das Formular nach dem Close nicht mehr benötigt wird bzw. nicht mehr vorhanden sein darf ist
Delphi-Quellcode:
wohl eine gute Lösung.
MyForm.Close; <<< eventuelle OnClose Routinen ausführen
FreeAndNil(MyForm); <<< nilen und Speicher freigeben |
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
Delphi-Quellcode:
frmTest := TfrmTest.Create(Application);
try frmTest.ShowModal; finally //was nun? frmTest.Release; //oder FreeAndNil(frmTest); end; |
Re: Warum macht ein TForm.Close nicht NIL
.Release ändert den Wert der Referenzvariable nicht!
|
Re: Warum macht ein TForm.Close nicht NIL
Release ist definitiv die bessere Lösung.
Und egal ob Close oder Release: man kann beides ohne ![]() |
Re: Warum macht ein TForm.Close nicht NIL
Ok, dann so:
Delphi-Quellcode:
Ich seh immer noch nicht so richtig den Unterschied, außer dass Release nicht wartet bis dass zu entfernende Fenster fertig mit allen Botschaften ist. Was passiert, wenn ich Release verwende um beim Beenden des Programmes die Unterfenster zu schließen? Das Release fängt an das Unterfenster zu entfernen und gibt die Kontrolle noch bevor es fertig ist wieder zurück an das Hauptprogramm. Dieses schließt sich nun selber. Was aber, wenn das Release des Unterfensters länger als das Beenden des Programmes dauert?
frmTest := TfrmTest.Create(Application);
try frmTest.ShowModal; finally //was nun? frmTest.Release; frmtest := nil; //oder FreeAndNil(frmTest); end; Theoretisches Beispiel:
Delphi-Quellcode:
@Muetze: meinst du das so?
frmTest := TfrmTest.Create(Application);
try frmTest.ShowModal; finally //was nun? frmTest.Release; //dauert 5 Sekunden - warum auch immer, aber laut Hilfe: Release gibt die Steuerung sofort an die aufrufende Routine zurück und wartet nicht, bis das Formular freigegeben wird. //Alles ab hier wird also ausgeführt bevor frmTest.Release überhaupt fertig ist. frmtest := nil; end; Close; //Programm selber beenden - was aber wenn frmTest.Release noch läuft?
Delphi-Quellcode:
procedure TfrmTest.FormDestroy(Sender: TObject);
begin Self := nil; //so? end; |
Re: Warum macht ein TForm.Close nicht NIL
Zitat:
|
Re: Warum macht ein TForm.Close nicht NIL
Danke für den Tipp. :thumb:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 21:50 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz