AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Speicherverwendung analysieren

Ein Thema von eike42 · begonnen am 27. Jun 2012 · letzter Beitrag vom 27. Jun 2012
Antwort Antwort
Seite 1 von 2  1 2      
eike42

Registriert seit: 9. Mär 2010
18 Beiträge
 
RAD-Studio 2009 Ent
 
#1

Speicherverwendung analysieren

  Alt 27. Jun 2012, 08:40
Hallo zusammen,

ich habe vor 2 Jahren ein ziemlich grosses Delphi-Projekt übernommen, ein Riesenframework mit ziemlich grosser Oberfläche und ziemlich vielen Bugs.

Mein größtes Problem ist im Moment ein Speicherproblem, d.h. der Arbeitsspeicher läuft voll und ich weiss nicht wieso. Speicherlecks gibt es auch (FastMM sei Dank), allerdings tritt das Speicherproblem schon beim Laden des Projekts auf, ist also höchstwahrscheinlich auf die Datenstruktur/Datenhaltung im Speicher zurückzuführen.

Das Projekt ist gar nicht so gross, serialisiert (XML) etwa 20-30 MB, also weit entfernt von 2GB, selbst wenn man den Delphi-Overhead einrechnet.

Ich würde jetzt gerne mal eine Übersicht der im Speicher befindlichen Objekte sehen, und zwar als Übersicht Klassenname/Anzahl/Speicherbedarf, damit ich weiss welche Objekte (wir haben über 1000) Amok laufen und wo ich beim Debuggen ansetzen muss.

Kennt jemand ein Programm das das kann? Kann ich FastMM dazu bringen, mir nicht die Memory-Leaks, sondern *alle* im Speicher befindlichen Objekte auszuspucken?

Ach ja, ich nutze Rad Studio 2009.

Danke und viele Grüße
Eike
Eike
  Mit Zitat antworten Zitat
hathor
(Gast)

n/a Beiträge
 
#2

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 09:58
How Much Memory Is Your Delphi Program Occupying? ... and more...
The TProcessMemoryCounters record wraps up the Windows API PROCESS_MEMORY_COUNTERS structure. Here's the meaning of the other fields:

PageFaultCount - the number of page faults.
PeakWorkingSetSize - the peak working set size, in bytes.
WorkingSetSize - the current working set size, in bytes.
QuotaPeakPagedPoolUsage - The peak paged pool usage, in bytes.
QuotaPagedPoolUsage - The current paged pool usage, in bytes.
QuotaPeakNonPagedPoolUsage - The peak nonpaged pool usage, in bytes.
QuotaNonPagedPoolUsage - The current nonpaged pool usage, in bytes.
PagefileUsage - The current space allocated for the pagefile, in bytes.
Those pages may or may not be in memory.
PeakPagefileUsage - The peak space allocated for the pagefile, in bytes.

http://delphi.about.com/od/delphitip...mory_usage.htm

Delphi-Quellcode:
uses PsAPI

function CurrentMemoryUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.WorkingSetSize
   else
     RaiseLastOSError;
 end;

 function PageFaultCount: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.PageFaultCount
   else
     RaiseLastOSError;
 end;

 function PeakWorkingSetSize: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.PeakWorkingSetSize
   else
     RaiseLastOSError;
 end;

 function WorkingSetSize: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.WorkingSetSize
   else
     RaiseLastOSError;
 end;

 function QuotaPeakPagedPoolUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.QuotaPeakPagedPoolUsage
   else
     RaiseLastOSError;
 end;

 function QuotaPagedPoolUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.QuotaPagedPoolUsage
   else
     RaiseLastOSError;
 end;

 function QuotaPeakNonPagedPoolUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.QuotaPeakNonPagedPoolUsage
   else
     RaiseLastOSError;
 end;

 function QuotaNonPagedPoolUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.QuotaNonPagedPoolUsage
   else
     RaiseLastOSError;
 end;

 function PagefileUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.PagefileUsage
   else
     RaiseLastOSError;
 end;

 function PeakPagefileUsage: Cardinal;
 var
   pmc: TProcessMemoryCounters;
 begin Result:=0;
   pmc.cb := SizeOf(pmc) ;
   if GetProcessMemoryInfo(GetCurrentProcess, @pmc, SizeOf(pmc)) then
     Result := pmc.PeakPagefileUsage
   else
     RaiseLastOSError;
 end;

procedure TForm1.Timer1Timer(Sender: TObject);
begin
Memo1.Clear;
Memo1.Lines.add(FormatFloat('Memory used: ,.# KB', CurrentMemoryUsage / 1024));
Memo1.Lines.add('PageFaultCount: '+ IntToStr(PageFaultCount));
Memo1.Lines.add(FormatFloat('PeakWorkingSetSize: ,.# KB', PeakWorkingSetSize / 1024));
Memo1.Lines.add(FormatFloat('WorkingSetSize: ,.# KB', WorkingSetSize / 1024));
Memo1.Lines.add(FormatFloat('QuotaPeakPagedPoolUsage: ,.# KB', QuotaPeakPagedPoolUsage / 1024));
Memo1.Lines.add(FormatFloat('QuotaPagedPoolUsage: ,.# KB', QuotaPagedPoolUsage / 1024));
Memo1.Lines.add(FormatFloat('QuotaPeakNonPagedPoolUsage: ,.# KB', QuotaPeakNonPagedPoolUsage / 1024));
Memo1.Lines.add(FormatFloat('QuotaNonPagedPoolUsage: ,.# KB', QuotaNonPagedPoolUsage / 1024));
Memo1.Lines.add(FormatFloat('PagefileUsage: ,.# KB', PagefileUsage / 1024));
Memo1.Lines.add(FormatFloat('PeakPagefileUsage: ,.# KB', PeakPagefileUsage / 1024));
end;

Geändert von hathor (27. Jun 2012 um 15:32 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu
Online

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.080 Beiträge
 
Delphi 12 Athens
 
#3

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 10:05
Praktisch gesehn und da es in Delphi (eigentlich) keine zentrale Stelle gibt, wo reservierter Speicher "registriert" wird, kann man im Nachhinein garnicht feststellen was sich in welchem Speicherbereich befindet.

Nimm dir einfach mal sowas wie Integer, Records oder ein Array[x..y] of Char. Da hast du nur den Daten-Inhalt im RAM rumliegen, aber nirgendwo ist die RTTI direkt damit verknüpft.
(z.B. 4 Byte im Speicher können also ganz gut ein Integer, 4 Bytes, 2 Words, ein Array[0..3] of AnsiChar oder auch garnichts sein, da diese 4 Bytes grade nicht benutzt werden)

Man kann höchstens über windige und nicht immer zuverlässige Tests versuchen rauszufinden, ob es sich z.B. um ein Objekt handet. Seit D2009 könnte man eventuell auch noch AnsiStrings und UnicodeStrings erkennen. (alles immer schön mit der Chance auf viele False-Positives)

Man könnte den FastMM in den FullDebug-Modus versetzen, oder sich selbst dazwischenhängen, dann jeweils beim reservieren von Speicher sich den Stacktrace, bzw. die Rücksprungadresse ansehn, und bekommt so mit wer und von wo die Anfrage kommt, worüber man besser erkennen kann, um was es sich handelt, da der Speicher von Strings und Objekten ja standardmäßig in ganz bestimmten Prozeduren/Methoden reserviert wird.
Wenn man sich diese infos dann alle merkt, dann könnte man auch später oftmals noch gut erkennen was von wo kam.
Es gibt da auch schon ein irgendwelche Frameworks, welche einem dabei helfen. (Ich glaub vor Kurzem war auch was zum Speicheranalysieren hier in der DP zu finden)
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.

Geändert von himitsu (27. Jun 2012 um 10:10 Uhr)
  Mit Zitat antworten Zitat
PMM

Registriert seit: 17. Feb 2005
101 Beiträge
 
#4

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 10:25
Als Tool könnte ich, wenn es was kosten darf, AutomatedQA/SmartBear AQTime empfehlen. Das kann auch "profilen" wo/wann genau und wie viel Speicher angefordert / freigegeben wurde.
PMM
  Mit Zitat antworten Zitat
sahimba

Registriert seit: 14. Nov 2011
Ort: Berlin, Hauptstadt der DDR
137 Beiträge
 
Delphi 10 Seattle Professional
 
#5

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 10:29
Du könntest hier mal schauen: http://ddobjects.de/ddserver.
Im Gegensatz zum FastMM zeigt Dir das Tool die Verwendung live an, was gerade dann vorteilhaft ist wenn es sich nicht um Leaks handelt sondern um wachsenden Speicherbedarf welcher bei Programmende sauber abgeräumt wird. Das Tool kann Objekte, Records, Arrays und Strings tracen, zeigt Dir die Callstacks an, unterstützt mehrere Threads als auch die Verwendung von Packages. Das eine oder andere Utility in dem Paket ermöglich auch bspw. die zu frühe Freigabe von Speicher zu erkennen (eine Exception im Destructor welche darauf beruht, dass ein Objekt "irgendwo" frühzeitig freigegeben worden ist kann mitunter ziemlich schwer zu beheben sein...), enthält einen (noch nicht sehr ausgereiften) Exceptionhandler sowie einen Threadviewer welcher Dir den Status (incl. Callstacks) der einzelnen Threads der Applikation anzeigt.

ACHTUNG! Die Version auf der Webseite ist veraltet! Ich überarbeite das Tool derzeit. Bei Interesse schickst Du mir bitte eine PN und ich kann es Dir in den nächsten Tagen zukommen lassen. Bin derzeit im Umzugsstress, kann also nicht sofort reagieren.

Grüße,
S.
  Mit Zitat antworten Zitat
eike42

Registriert seit: 9. Mär 2010
18 Beiträge
 
RAD-Studio 2009 Ent
 
#6

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 13:44
Hi, schon mal danke für die Antworten.

@sahimba, ich werde später auf dich zurückkommen. Bin im Moment noch mit anderen Dingen beschäftigt und muss erstmal Zeit für so eine Analyse freiräumen.

Praktisch gesehn und da es in Delphi (eigentlich) keine zentrale Stelle gibt, wo reservierter Speicher "registriert" wird, kann man im Nachhinein garnicht feststellen was sich in welchem Speicherbereich befindet.
Genau deshalb hatte ich gehofft dass es einen Profiler gibt, der schon bei der Erstellung der Objekte "mitzählt". Ggf. müsste ich das über unsere ObjectFactory(s) machen, dann weiss ich allerdings nicht wieviele "direkt" erstellte Objekte mir entgehen. Wie gesagt, es geht hier um über 1000 verschiedene Klassen, und deshalb hilft mir eigentlich nur der Klassenname weiter.
Eike
  Mit Zitat antworten Zitat
sahimba

Registriert seit: 14. Nov 2011
Ort: Berlin, Hauptstadt der DDR
137 Beiträge
 
Delphi 10 Seattle Professional
 
#7

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 14:01
Genau deshalb hatte ich gehofft dass es einen Profiler gibt, der schon bei der Erstellung der Objekte "mitzählt".
Genau das tut mein Profiler inkl. einer weiteren Übersicht, welche die aktuelle Anzahl der Objekte einer jeden Klasse anzeigt, wie viele maximal zu gleicher Zeit erzeugt waren sowie die Gesamtanzahl aller erzeugten/freigegebenen Objekte. Auch auf dem Wege lässt sich relativ einfach ein wachsender Speicherbedarf zur Laufzeit erkennen und gezielt beheben.

Melde Dich dann, sobald Du Zeit freigeräumt hast.

Grüße,
S.
  Mit Zitat antworten Zitat
eike42

Registriert seit: 9. Mär 2010
18 Beiträge
 
RAD-Studio 2009 Ent
 
#8

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 14:41
Genau deshalb hatte ich gehofft dass es einen Profiler gibt, der schon bei der Erstellung der Objekte "mitzählt".
Genau das tut mein Profiler inkl. einer weiteren Übersicht, welche die aktuelle Anzahl der Objekte einer jeden Klasse anzeigt, wie viele maximal zu gleicher Zeit erzeugt waren sowie die Gesamtanzahl aller erzeugten/freigegebenen Objekte. Auch auf dem Wege lässt sich relativ einfach ein wachsender Speicherbedarf zur Laufzeit erkennen und gezielt beheben.

Melde Dich dann, sobald Du Zeit freigeräumt hast.

Grüße,
S.
Wow, das habe ich aus deiner Webseite (sorry) nicht herauslesen können.

Danke und Gruss
Eike
Eike
  Mit Zitat antworten Zitat
Benutzerbild von DeddyH
DeddyH

Registriert seit: 17. Sep 2006
Ort: Barchfeld
27.624 Beiträge
 
Delphi 12 Athens
 
#9

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 14:44
Also, daran hätte ich auch Interesse.

P.S.: Ist das das Tool, das Du in Berlin auf den DT erwähnt hast?
Detlef
"Ich habe Angst vor dem Tag, an dem die Technologie unsere menschlichen Interaktionen übertrumpft. Die Welt wird eine Generation von Idioten bekommen." (Albert Einstein)
Dieser Tag ist längst gekommen
  Mit Zitat antworten Zitat
sahimba

Registriert seit: 14. Nov 2011
Ort: Berlin, Hauptstadt der DDR
137 Beiträge
 
Delphi 10 Seattle Professional
 
#10

AW: Speicherverwendung analysieren

  Alt 27. Jun 2012, 14:46
P.S.: Ist das das Tool, das Du in Berlin auf den DT erwähnt hast?
Yup, das ist es. Schon eine ganze Weile her, das war 2011...
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 00:26 Uhr.
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz