![]() |
Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Hallo,
in einer sehr komplexen Anwendung habe ich bei einer Klasse ein Interface entfernt. Beispiel:
Delphi-Quellcode:
Ich habe also IBla4 entfernt. Es wird auch nicht benötigt. Alles kompiliert. Alle tausende Tests (Unit-Tests und UI-Tests) laufen korrekt durch ausser einem. Jener bemängelt, dass die Anwendung mehr Speicher benötigt. Nicht viel, aber doch über der üblichen Schwankung.
//vorher
TBla = class(TVorfahr, IBla1, IBla2, IBla3, IBla4) //nachher TBla = class(TVorfahr, IBla1, IBla2, IBla3) Hat wer eine Ahnung woran das liegen könnte? Kann sich jemand ein Scenario vorstellen wie so was passieren kann? (In anderen Fällen gab es schon mal eine Kreuzreferenz was dann durch "[weak]" oder Umstellung gelöst werden konnte. Aber sowas kann ich mir nicht vorstellen.) |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Hi,
I think, even if IBla4 wasn't needed but its functions is being called, may be one of them return a reference counted class/object/interface, if such function being called from the object TBla directly then it will be handled as usual class function, but if it was called from something like IBla4 within TBla then reference counted will be invoked and memory will be cleaned earlier, not leaving it hanging there waiting the application to close. |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Hast du wirklich nur in der Klassendefinition das Interface entfernt? Und wenn du es wieder hinzufügt, ist der Speicherverbrauch wieder normal?
Um welche Werte reden wir denn hier? Wie oft liegt die Klasse im Speicher und wie hoch sind dabei die Unterschiede im Verbrauch in absoluten Werten? |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Hallo,
ja ich habe nur diese id und das Komma entfernt. Die Werte sind so bei 3%. Das ist nicht recht viel. Dafür dass diese Klasse eine von 24 ist und nur dieses interface weg bekommt recht viel. Wie oft das tatsächlich ist und wie die absoluten Werte sind kann ich nicht sagen. Das genauer zu untersuchen war mir bislang zuviel Aufwand. Ein Fall ist mir noch eingefallen: if not Supports(Klasse, IBla4) then ProduziereSpeicherLeck(); :-D Solche "if not Supports" kommen tatsächlich vor. Sie dürften jedoch nie aufgerufen werden und führen nach then zu einem raise einer Exception. Nun denn, mal schauen wie ich da weiter mache. Auf jeden Fall nicht mehr dieses Jahr :wink: Danke für Eure Denkanstösse. |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Mehr Speicherverbrauch 3% von was? Der gesamten Anwendung, der Klasse oder ...?
Unterstützt TVorfahr bereits IBla4? Unterstützt ein Nachfahre von TBla IBLa4? Jeder Nachfahren von TBLa der zusätzlich IBLa4 unterstützen, braucht eine eigene Liste von Interfaces. Stimmt die Liste mit der von TBLa überein, wird vieleicht die selbe Liste verwendet. Das könnte zumindest einen kleinen Unterschied im Speicherbedarf für die Klassen erklären. |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Zitat:
Wenn es z.B. 100.000 Nachfahr-Objekte gibt, macht das einiges aus. Ist es nur eine einzelne Instanz, würde das nicht passen. |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Also... soweit ich das sehe (bitte korrigieren, falls ich falsch liege..:), wird für jedes Interface ein zusätzliches Feld im
Ojekt angelegt. (siehe TObject.InitInstance in system.pas) . D.h. Wenn das Interface hinzugefügt wird, erhöht sich der Speicher dementsprechend um 4Byte pro angelegtes Ojekt. Man kann sich den Speicherverbrauch einer Klasse/Objekt auch per "InstanceSize" anschauen....
Delphi-Quellcode:
unit Unit1;
interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type IIntf1 = interface ['{6AB3BE8C-8605-4ED6-97DA-CEED84E5B9F4}'] procedure Test1; end; IIntf2 = interface ['{A2614303-1914-4BBE-9D38-D306EF36B14B}'] procedure Test2; end; type TObj1 = class(TInterfacedObject, IIntf1) public procedure Test1; procedure Test2; end; type TObj2 = class(TInterfacedObject, IIntf1, IIntf2) public procedure Test1; procedure Test2; end; type TForm1 = class(TForm) Button1: TButton; procedure Button1Click(Sender: TObject); private { Private-Deklarationen } public { Public-Deklarationen } end; var Form1: TForm1; implementation {$R *.dfm} procedure TForm1.Button1Click(Sender: TObject); var obj1 : TObj1; obj2 : TObj2; begin obj1 := TObj1.Create; obj1.Test1; obj1.Free; obj2 := TObj2.Create; obj2.Test1; obj2.Free; end; { TObj1 } procedure TObj1.Test1; begin ShowMessage('Obj1 size:' + IntToStr(InstanceSize)); end; procedure TObj1.Test2; begin ShowMessage('obj1 test2'); end; { TObj2 } procedure TObj2.Test1; begin ShowMessage('Obj2 size: ' + intToStr(InstanceSize)); end; procedure TObj2.Test2; begin ShowMessage('test2'); end; end. |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Ein Interface zeigt auf eine Interfacetabelle, welche in der RTTI neben dem Objekt abgelegt wird, also ja, es kommt ein "Pointer" hinzu.
|
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Zitat:
Obj1 und Obj2 müssen vom Interface type sein und natürlich nicht mit free freigegeben werden. Mavarik :coder: |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Natürlich ist das nur ein schnell runter getipptes Ding - und natürlich würde ich interfaces so nicht verwenden...
Die Mixtur kann schnell in die Hose gehen ;) Aber da ich hier ja nur in der "Objektwelt" bin muss ich ja das Ojbekt brav freigeben (solange ich die Interfaces nicht anrühre ;) und das RefCounting initiere ). |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Hallo zusammen,
ich bin wieder dran. Einige Antworten/Bemerkungen: Die 3% sind für die gesammte Anwendung. Dass für jedes Interface ein zusätzliches Feld im Ojekt angelegt wird finde ich plausibel. Dann müsste die Anwendung jedoch mit interface mehr Speicher benötigen als ohne. Das umgekehrte ist jedoch der Fall. Unterstützt TVorfahr bereits IBla4? --> Nein, aber selbst dann dürfte ein Entfernen nicht mehr Speicher benötigen. Ein Vorfahr unterstützt jedoch noch ein interfaces. Unterstützt ein Nachfahre von TBla IBLa4? --> Es gibt keine Nachfahren Inzwischen habe ich noch einen Fall von if ... Supports(...) gefunden und dabei bemerkt, dass ich etwas benötige von dem interface. (Nicht alles drum habe ich das nun aufgeteilt.) Ich kann also nicht ganz darauf verzichten auf ein Interface. Die Frage ist daher nun naheuu obsolet. Merkwürdig bleibt es trotzdem. |
AW: Klasse implementiert ein Interface weniger, höherer Speicherverbrauch
Schade, solche Mysterien sind eigentlich immer spannend.
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 19:57 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