![]() |
Speicherleaks TMemoryStream in einem Objekt
Hallo Zusammen,
ich habe meine Client-Server App massiv umgebaut und dabei soviele Speicherleaks geschaffen, dass die Anwendung wegen überfülltem Speicher nach 1-2 Stunden abstürzt. Allerdings finde ich den Fehler nicht. Ich versuche mich gerade in FastMM4 einzuarbeiten. Aber mal eine grundsätzliche Frage: Werden TMemoryStreams, die ich in einem Objekt definiere, freigegeben, wenn ich das Objekt freigebe? Beispiel: Definition
Delphi-Quellcode:
Aufruf aus der Form
Type
TMxSQL = class fColsSetMain: TCols; fRowsSetMain: TRows; fStreamSetMain: TMemoryStream; fColsSetBSC: TCols; fRowsSetBSC: TRows; fStreamSetBSC: TMemoryStream; private { Private-Deklarationen } public { Public-Deklarationen } property ColsSetMain: TCols read fColsSetMain write fColsSetMain; property RowsSetMain: TRows read fRowsSetMain write fRowsSetMain; property ColsSetBSC: TCols read fColsSetBSC write fColsSetBSC; property RowsSetBSC: TRows read fRowsSetBSC write fRowsSetBSC; property StreamSetMain: TMemoryStream read fStreamSetMain write fStreamSetMain; property StreamSetBSC: TMemoryStream read fStreamSetBSC write fStreamSetBSC; constructor Create(GetSets: boolean = true);
Delphi-Quellcode:
Der Constructor
procedure TFrm_Main_BSC.FormShow(Sender: TObject);
var I: integer; SpecialStart: boolean; MxSQL: TMxSQL; begin Try MxSQL:= TMxSQL.Create; Try ... Finally MxSQL.Free; End;
Delphi-Quellcode:
Die Datenbank-Abfrage:
constructor TMxSQL.Create(GetSets: boolean);
begin if GetSets then begin fStreamSetMain:= TMemoryStream.Create; GetSettings('hlp_properties'); fStreamSetBSC:= TMemoryStream.Create; GetSettings('hlp_properties_bsc'); end; end;
Delphi-Quellcode:
Hierbei passiert irgendwo neben vielen anderen Stellen eine Speicherleak.
procedure TMxSQL.GetSettings(Tabelle: string);
var LClient: TxDataClient; LService: IDBService; Logic: TLogic; begin LClient := TXDataClient.Create; Logic:= TLogic.Create; Try LClient.Uri:= DB_Unit.xData_Connect.URL; LService:= LClient.Service<IDBService>; if Tabelle = 'hlp_properties_bsc' then begin fStreamSetBSC:=LService.Properties_BSC_Select as TMemoryStream; fStreamSetBSC.Position:=0; Logic.StreamToRows(fStreamSetBSC, fColsSetBSC, fRowsSetBSC); end else if Tabelle = 'hlp_properties' then begin fStreamSetMain:=LService.Properties_Select as TMemoryStream; fStreamSetMain.Position:= 0; Logic.StreamToRows(fStreamSetMain, fColsSetMain, fRowsSetMain); end; Finally LClient.Free; Logic.Free; end; end; Das ist die Zusammenfassung aus dem Report von FastMM4: Zitat:
Vielen Dank Patrick |
AW: Speicherleaks TMemoryStream in einem Objekt
Du erstellst etwas im Constructor, was sich nicht selbst freigibt,
warum fehlt dann dein Destructor? Wo ist das
Delphi-Quellcode:
?
inherited Create;
Und wieso gibt es ein WRITE bei den Streams? Die werden intern erstellt, also was wird wohl passieren, wenn dort wirklich mal jemand etwas zuweist? |
AW: Speicherleaks TMemoryStream in einem Objekt
Ok, da hast Du direkt die Lücken getroffen...:oops:
Zitat:
Delphi-Quellcode:
Ich nehme an, die korrekte Definition des Contructors sähe so aus:
property StreamSetMain: TMemoryStream read fStreamSetMain;
property StreamSetBSC: TMemoryStream read fStreamSetBSC;
Delphi-Quellcode:
Und dann müsste ich den Destructor auch anpassen, oder?
constructor TMxSQL.Create(GetSets: boolean);
begin inherited Create; if GetSets then begin fStreamSetMain:= TMemoryStream.Create; GetSettings('hlp_properties'); fStreamSetBSC:= TMemoryStream.Create; GetSettings('hlp_properties_bsc'); end; end;
Delphi-Quellcode:
Aber wenn ich es so mache, crasht die App sofort - das scheint nicht der richtige Weg zu sein...
destructor TMxSQL.Free;
begin fStreamSetMain.Free; fStreamSetBSC.Free; Inherited Free; end; Hast Du einen Tip für mich? |
AW: Speicherleaks TMemoryStream in einem Objekt
Der Destructor muss
Delphi-Quellcode:
heißen, nicht Free.
Destroy
|
AW: Speicherleaks TMemoryStream in einem Objekt
Hallo,
im Constructor wird der TMemoryStream nicht immer erzeugt if GetSets then begin fStreamSetMain:= TMemoryStream.Create; in Destructor aber immer freigegeben fStreamSetMain.Free; Besser wäre FreeAndNil(fStreamSetMain) |
AW: Speicherleaks TMemoryStream in einem Objekt
Ich habe es jetzt so angepasst und es gibt zumindest keinen Crash mehr...
Delphi-Quellcode:
Aufruf:
destructor Destroy; override;
... destructor TMxSQL.Destroy ; begin FreeAndNil(fStreamSetMain); FreeAndNil(fStreamSetBSC); SetLength(fColsSetMain, 0); SetLength(fRowsSetMain, 0, 0); SetLength(fColsSetBSC, 0); SetLength(fRowsSetBSC, 0, 0); Inherited; end;
Delphi-Quellcode:
MxSQL.Free;
Aber das Speicherleak ist immernoch katastrophal... Zitat:
|
AW: Speicherleaks TMemoryStream in einem Objekt
Hallo,
naja, Katastrophe ist es nicht. ;) Vorschlag Definiere abgeleitete Klassen für Deine TMemoryStreams. Also so:
Delphi-Quellcode:
und dann
TMemoryStream_StreamSetMain = class(TMemoryStream)</Delphi>
Delphi-Quellcode:
usw.
fStreamSetMain:= TMemoryStream_StreamSetMain.Create;
Das schöne ist, Du siehst im FastMM4-Log den Namen des MemStreams, der erzeugt, und nicht freigegeben wird. PS: wird nicht sogar die Zeile angezeigt, wo der Stream erzeugt wurde??? |
AW: Speicherleaks TMemoryStream in einem Objekt
Zitat:
Der Vorschlag mit der abgeleitet Klasse ist glaube ich gar nicht nötig, denn ich kann in der Speicherbeschreibung sehen, um welche Tabelle es sich handelt: Zitat:
Aktuell habe ich noch das Problem, dass mit dem angepasst Destructor die Applikation bei einer MutiThread Procedure abstüzt. Habe sie schon angepasst, aber ohne Erfolg:
Delphi-Quellcode:
constructor TMxSQL.Create(GetSets: boolean);
begin inherited Create; fStreamCreated:= false; if GetSets then begin fStreamCreated:= true; fStreamSetMain:= TMemoryStream.Create; GetSettings('hlp_properties'); fStreamSetBSC:= TMemoryStream.Create; GetSettings('hlp_properties_bsc'); end; end; destructor TMxSQL.Destroy ; begin if fStreamCreated then begin if fStreamSetMain <> nil then FreeAndNil(fStreamSetMain); if fStreamSetBSC <> nil then FreeAndNil(fStreamSetBSC); SetLength(fColsSetMain, 0); SetLength(fRowsSetMain, 0, 0); SetLength(fColsSetBSC, 0); SetLength(fRowsSetBSC, 0, 0); end; Inherited; end; Zitat:
|
AW: Speicherleaks TMemoryStream in einem Objekt
Vergesst des Crash - habe den Fehler gefunden...
Hatte gestern testweise die Streams manuell freigegebe:
Delphi-Quellcode:
Dann muss es mit dem Destructor ja crashen...
MxSQL.StreamSetMain.Free;
MxSQL.StreamSetBSC.Free; MxSQL.Free; |
AW: Speicherleaks TMemoryStream in einem Objekt
So, ich habe jetzt mal ein bißchen weiter ausprobiert und komme zu einer Frage, bei der ich Euch wahrscheinlich bis hierhin lachen höre:duck:
Wenn ich eine Function habe, die als Rückgabewert einen Stream hat, wird dann der Streaminhalt oder nur der Pointer zurückgegeben? Ich habe bislang in der aufrufenden Procedure immer eine Variable vom Type T(Memory)Stream created, ihr die Funktion zugewiesen und am Ende der Procedure freigegeben. Aber wenn nur der Pointer übergeben wird, darf ich den Stream in der aufrufenden Procedure gar nicht createn, oder? Der würde ja dann im nirgendwo verschwinden... Und dann wäre e logisch, warum ich den Stream, obwohl an Result übergeben, nicht free setzen darf. Ist das so, dann muss ich einiges überdenken... LG Patrick |
Alle Zeitangaben in WEZ +1. Es ist jetzt 06:58 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 by Thomas Breitkreuz