Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   GUI-Design mit VCL / FireMonkey / Common Controls (https://www.delphipraxis.net/18-gui-design-mit-vcl-firemonkey-common-controls/)
-   -   Delphi Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML (https://www.delphipraxis.net/182283-ungeheure-speichernutzung-des-twebrowsers-beim-laden-einer-xml.html)

Ajintaro 15. Okt 2014 09:55

Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Hallo DP !

Ich habe eine sehr simple Anwendung erstellt mit nur 4 Komponenten:

- TWebbrowser
- TButton
- TStatusBar
- TTimer

Eine Funktion zur Ermittlung der Hauptspeichernutzung des eigenen Programms:

Delphi-Quellcode:
function getMyMemoryUsageInKb : integer;
var
  pmc: PPROCESS_MEMORY_COUNTERS;
  cb: Integer;
begin
 cb := SizeOf(_PROCESS_MEMORY_COUNTERS);
 GetMem(pmc, cb);
 pmc^.cb := cb;
 if(GetProcessMemoryInfo(GetCurrentProcess, pmc, cb))then
   result := pmc^.WorkingSetSize div 1024
 else
   result := -1;
 FreeMem(pmc);
end
Ich navigiere per button zu meiner XML Datei (ca. 5,6 MB):

Delphi-Quellcode:
webbrowser1.Navigate('http://www.dragonlords.info/dev/big.xml');
Der Timer zeigt mir nur die RAM-Auslastung an:

Delphi-Quellcode:
Statusbar1.Panels[0].Text := 'RAM: '+ FLoatToStr(Ceil(getMyMemoryUsageInKb/1024)) + ' MB';
Mehr macht diese Anwendung nicht. Das erschreckende daran ist, sie nutzt ca. 750-800 MB Hauptspeicher !

Ich möchte aber mehrere XML Dateien ähnlicher Größenordnung nacheinander untersuchen, jedoch schmiert mir die Entwickler-VM aufgrund zu wenig Hauptspeicher ab. Der IE 11 des Betriebssystems addressiert bei diesem Seitenaufruf ca. 220MB und wächst in kleinen Schritten bis ... weiß nicht genau, bei 280MB habe ich ihn geschlossen.

Wie kann man denn solche großen XML files untersuchen ohne den RAM Bedarf der eigenen Anwendung auf Gigabytegröße anwachsen zu lassen ?

www.dragonlords.info/dev/leak_demo.rar

Der schöne Günther 15. Okt 2014 10:06

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Grade mal ausprobiert. Ja, wird riesig. Warum? Keine Ahnung.

Aber warum in aller Welt will man das tun? Die XML in einem TWebBrowser anzuzeigen wird ja nicht dein eigentliches Ziel sein. Wenn du nur zu Debugzwecken den Text irgendwo haben willst, nimm doch ein TMemo.

Willst du die XML auswerten, mach das doch über
Delphi-Quellcode:
Xml.XmlIntf.IXMLDocument
. Das braucht dann keine 20MB Hauptspeicher (zumindest bei mir).


Hey, Sankt Augustin.

Ajintaro 15. Okt 2014 10:33

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Hi,

ich muss das XML nicht im Browser darstellen, ich brauch das reine XML in einem TMemo. Das hab ich folgendermaßen gebaut:

WebBrowser1DocumentComplete:

Delphi-Quellcode:
var
xml:WideSstring;
doc: OleVariant;

doc := Webbrowser1.Document;
xml := doc.XMLDocument.xml;
Memo1.lines.add(xml);

Der schöne Günther 15. Okt 2014 11:00

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Den TWebbrowser habe ich nie benutzt, lehne mich aber soweit aus dem Fenster zu behaupten dass der vollkommen fehl am Platz ist.
Wenn du die XML von einer URL laden willst kannst du
  • die Indy-Komponenten verwenden
  • Das XML direkt mittels LoadXmlDocument(..) laden
  • Bestimmt noch anders

Zu 2):
Delphi-Quellcode:
unit Unit2;

interface

uses
   System.Classes,
   Vcl.Forms, Vcl.StdCtrls, Vcl.Controls,
   Xml.XmlIntf;

type
   TForm2 = class(TForm)
      Memo1: TMemo;
      Button1: TButton;
      Button2: TButton;

      procedure Button1Click(Sender: TObject);
      procedure Button2Click(Sender: TObject);

      protected var
         myXmlDocument: IXMLDocument;
   end;

var
   Form2: TForm2;

implementation uses Xml.XmlDoc;

{$R *.dfm}

procedure TForm2.Button1Click(Sender: TObject);
begin
   myXmlDocument := LoadXMLDocument('http://www.dragonlords.info/dev/big.xml');
end;

procedure TForm2.Button2Click(Sender: TObject);
begin
   Memo1.Lines.Assign(myXmlDocument.XML);
end;

end.

Das Einfügen in die Memo dauert ewig. Ich denke das liegt daran, dass deine 5MB-Datei nur aus einer einzigen Zeile besteht. Selbst Notepad++ bekommt hier Performance-Probleme. Das habe ich bei Notepad++ noch nie gesehen!

Spendier der XML doch Zeilenumbrüche.

himitsu 15. Okt 2014 12:22

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von Ajintaro (Beitrag 1275980)
ich muss das XML nicht im Browser darstellen, ich brauch das reine XML in einem TMemo. Das hab ich folgendermaßen gebaut:

Und warum nimmst du dann überhaupt einen Browser?

Delphi-Referenz durchsuchenTIdHTTP :roll:



Ich will zur Pizzabude gegenüber und statt den Roller nehm ich lieber einen 70-Tonner, mit großer Ladefläsche für die Pizza.

Mike_on_Tour 15. Okt 2014 12:27

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Hallo,

ich musste gerade etwas schmunzeln, als ich die Überschrift gelesen habe. Ich kämpf(t)e auch mit den XML-Dateien und bin schon mehrfach an 3 GB Hauptspeicherverbrauch 'gestorben'. Wir verwenden teilweise eigene XML-Routinen, teilweise abgeleitet von ThaXML und teilweise abgeleitet von XMLParser (von Stefan Heymann). Nach ein paar Korrekturen im Sourcecode liegt der Speicherverbrauch unterhalb von 2 GB.

Mein Programm:
- erzeugt 4 bis 5 XML-Dateien mit jeweils über 780.000 Zeilen (mit Zeilenumbrüchen !)
- jede XML-Datei hat eine Größe von ca. 65 MB
- das erzeugen einer XML-Datei dauert zur Zeit etwa 40 Minuten (alles im Hauptspeicher)
- jede XML-Datei in eine Textdatei konvertieren
- die Textdatei hat eine Größe von ca. 5 MB
- das erstellen der Textdatei dauert ca. 8 Minuten

Die Anzeige einer XML-Datei habe ich mit VirtualTreeView (von Mike Lischke) realisiert. Das Lesen einer XML-Datei dauert hier ca. 15 Sekunden.

Zum Vergleich: das Lesen einer XML-Datei mit Notepad++ dauert ca. 30 Sekunden.

Mike

Sherlock 15. Okt 2014 12:41

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Das sind absurde Zahlen! XML ist doch nix anderes als "gepimpter" Text. Wer braucht da 40 Minuten zum erzeugen so einer popligen Datei? Bisher habe ich die XML-Komponentenumschifft und hab meine XML naiv händisch aufgebaut (jedes Tag ist eine eigene Methode, die für den wohlgeformten Ausdruck sorgt), ich komme in ähnliche Größenordnungen, dürfte mir aber keine 40 Minuten zum Aufbau erlauben.

Sherlock

Der schöne Günther 15. Okt 2014 12:44

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von himitsu (Beitrag 1276009)
Ich will zur Pizzabude gegenüber und statt den Roller nehm ich lieber einen 70-Tonner, mit großer Ladefläsche für die Pizza.

Think Big.

mjustin 15. Okt 2014 12:49

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von Ajintaro (Beitrag 1275968)

Wie kann man denn solche großen XML files untersuchen ohne den RAM Bedarf der eigenen Anwendung auf Gigabytegröße anwachsen zu lassen ?

Bei großen XML Dateen kann man Libraries mit einer "Streaming API" verwenden. Diese laden das Dokument nicht als DOM sondern liefern einen Stream von Elementen und Attributen, auf die man dann in Handlern reagieren kann (zum Beispiel Werte auslesen und summieren, oder nur die interessanten Wertte in eine eigene Struktur kopieren).

Für Delphi gibt es einige Implementierungen - MSXML und das recht neue aber laut Angaben performante OXML sind zwei Beispiele.

Mike_on_Tour 15. Okt 2014 13:00

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von Sherlock (Beitrag 1276018)
Das sind absurde Zahlen! ... Wer braucht da 40 Minuten zum erzeugen so einer popligen Datei? ... dürfte mir aber keine 40 Minuten zum Aufbau erlauben.

Ich habe mit Absicht weggelassen, was alles im Programm abläuft und woher die Daten kommen. Ich kann aber versichern, dass der Ablauf so schon seine Ordnung hat. Hätte ich eine bessere Lösung gehabt, wäre sie auch umgesetzt worden. Übrigens, dem Anwender dürften die ca. 4 Stunden "Aufwand" (5 x 40 Minuten + 5 x 8 Minuten) immer noch besser gefallen, als wie ca. 15000 Seiten zu drucken und zu sortieren.

Mike

Ajintaro 15. Okt 2014 13:04

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Hi,

Gute Neuigkeiten !
Ich habe die Browser-Komponente entfernt und wie von himitsu vorgeschlagen idHTTP genommen:

Delphi-Quellcode:
function URLEncode(const ASrc: string): string;
var
  i: Integer;
const
  UnsafeChars = ['*', '#', '%', '<', '>', ' ','[',']']; {do not localize}
begin
  Result := '';   {Do not Localize}
  for i := 1 to Length(ASrc) do
  begin
    if (ASrc[i] in UnsafeChars) or (not (ord(ASrc[i])in [33..128])) then
    begin {do not localize}
      Result := Result + '%' + IntToHex(Ord(ASrc[i]), 2); {do not localize}
    end
    else
    begin
      Result := Result + ASrc[i];
    end;
  end;
end;


procedure TForm4.Button1Click(Sender: TObject);
var
  aStream: TMemoryStream;
  Params: TStringStream;
begin
  button1.Caption:='please wait';
  aStream := TMemoryStream.create;
  Params := TStringStream.create('');
  try
    with IdHTTP1 do
    begin
      //Params.WriteString(URLEncode('user=' + Edit1.Text + '&'));
      //Params.WriteString(URLEncode('pass=' + Edit2.Text));
      // You can change headers like this:
      Request.ContentType := 'application/x-www-form-urlencoded';
      try
        Response.KeepAlive := False;
        Post('http://www.dragonlords.info/dev/big.xml', Params, aStream);
      except
        on E: Exception do
          showmessage('Error encountered during POST: ' + E.Message);
      end;
    end;
    aStream.WriteBuffer(#0' ', 1);
    aStream.Position := 0;
    Memo1.Lines.LoadFromStream(aStream);
    button1.Caption:='done';
  except
    aStream.Free;
    Params.Free;
  end;
end;
Die Ladezeit ist die selbe, aber diese Aktion "kostet" gerade mal 15MB !

himitsu 15. Okt 2014 13:08

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Es kommt immer darauf an, was man machen will und welche Funktionen man nutzt.
  • HTML-Browser zum Laden und Darstellen der XML, welcher das natürlich für die Anzeige in ein HTML-DOM läd und in eine aufwändige GraphicEngine überführt. (und zusätzlich noch ein XML-DOM, da es so bei ihm integriert ist, anstatt es als "einfache" Textdatei zu behandeln)
  • ein XML-DOM, welcher die Datei samt Metainformationen fein aufgedröselt im RAM verwaltet
  • ein XML-SAX, welches, während der Behandlung, nur den aktuellen Knoten und maximal den aktuellen Pfad im RAM hat und den Rest als Text-Stream irgendwo besitzt

Wenn man viele Knoten sequentiell verarbeiten will, dann wäre das Mittel der Wahl ein SAXParser/SAXWriter.

Und wnen man die XML-Datei nur runterladen will, dann nimmt man natprlich nur eine "Download"-Komponente und nicht gleich einen kompletten Browser.

Sherlock 15. Okt 2014 13:13

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von Mike_on_Tour (Beitrag 1276024)
Ich habe mit Absicht weggelassen, was alles im Programm abläuft und woher die Daten kommen. Ich kann aber versichern, dass der Ablauf so schon seine Ordnung hat.

Dann bin ich ja beruhigt, das es nicht um den reinen XML-Aufbau geht. Denn ich muss mich wohl oder übel derzeit auf WSDL/XSD-Import einlassen, und habe dabei gar kein so gutes Gefühl.

Sherlock

Mike_on_Tour 15. Okt 2014 13:25

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von Sherlock (Beitrag 1276030)
Zitat:

Zitat von Mike_on_Tour (Beitrag 1276024)
Ich habe mit Absicht weggelassen, was alles im Programm abläuft und woher die Daten kommen. Ich kann aber versichern, dass der Ablauf so schon seine Ordnung hat.

Dann bin ich ja beruhigt, das es nicht um den reinen XML-Aufbau geht. Denn ich muss mich wohl oder übel derzeit auf WSDL/XSD-Import einlassen, und habe dabei gar kein so gutes Gefühl.

Ich habe noch mal nachgerechnet. Die Erstellung der XML-Datei dauert tatsächlich nur ca. 5 Minuten. Die restliche Zeit wird für Berechnungen und Prüfungen verwendet. Meine 40 Minuten waren da wohl nicht korrekt.

Mike

Sherlock 15. Okt 2014 13:26

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Du Schelm, Du! Kannst doch nem alten Mann nicht so einen Schrecken einjagen.

Sherlock

Ajintaro 15. Okt 2014 13:56

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Hi nochmal,

Wie setzt man denn mit idHTTP einen Post Befehl mit mehreren Parametern ab ? Muss eigens dafür eine for - Schleife geschrieben werden um jeden Parameter einzeln:

Delphi-Quellcode:
Params.WriteString(URLEncode(KEY + VALUE + '&'));
zu encoden ?

Meine Parameter kommen in einer Zeile wie z.B. username=Rolf&pass=Geheim&Anzug=Gruen&Ort=SanktAug ustin
Muss eine Prozedur geschrieben werden um die einzelnen Wertepaare zu ermitteln?

Delphi-Quellcode:
{++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//submit_post
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
Procedure submit_post(url_string,parameter:string;amemo:TMemo);
var
  aStream: TMemoryStream;
  Params: TStringStream;
begin
  aStream := TMemoryStream.create;
  Params := TStringStream.create('');
  try
    with Fmain.IdHTTP1 do
    begin
       
      //Params.WriteString(URLEncode('user=' + Edit1.Text + '&'));
      //Params.WriteString(URLEncode('pass=' + Edit2.Text));

      Request.ContentType := 'application/x-www-form-urlencoded';
      try
        Response.KeepAlive := False;
        Post(url_string, Params, aStream);
      except
        on E: Exception do
          showmessage('Error encountered during POST: ' + E.Message);
      end;
    end;
    aStream.WriteBuffer(#0' ', 1);
    aStream.Position := 0;
    amemo.Lines.LoadFromStream(aStream);
  except
    aStream.Free;
    Params.Free;
  end;
end;

hathor 15. Okt 2014 14:28

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
#11 ist "umständlich"...

Besser so:

Delphi-Quellcode:
procedure TForm4.Button2Click(Sender: TObject);  //SAVE XML
var fs: TFileStream; XMLadress: String;
begin
  fs := TFileStream.Create('D:\big.xml', fmCreate);
  XMLadress:='http://www.dragonlords.info/dev/big.xml';
  try
      IdHTTP1.Get(XMLadress, fs);
  finally
    fs.Free;
  end;
end;

procedure TForm4.Button3Click(Sender: TObject);  //LOAD XML
begin
Memo1.Lines.LoadFromFile('D:\big.xml');
end;

Ajintaro 15. Okt 2014 14:42

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Danke für deinen Beitrag hathor, aber ich bekomme die xml nur durch POST und einen Schwall an Parametern.

mjustin 15. Okt 2014 20:20

AW: Ungeheure Speichernutzung des TWebrowsers beim Laden einer XML
 
Zitat:

Zitat von Ajintaro (Beitrag 1276039)
Wie setzt man denn mit idHTTP einen Post Befehl mit mehreren Parametern ab ?

Nach diesem Stackoverflow Beitrag kann man es ab Delphi 2009 so machen:

Delphi-Quellcode:
var
  Server: TIdHttp;
  Parameters,Response: TStringStream;
begin
  Response := TStringStream.Create;
  Parameters := TStringStream.Create('param1=Value1&param2=&#1593;&#1585;&#1576;&#1610;/&#1593;&#1585;&#1576;&param3=Value3', TEncoding.UTF8);
  Server.Post('http://www.example.com/page.php', Parameters, Response);
end;
Den Zeichensatz sollte man vor dem Post auch noch setzen:

Delphi-Quellcode:
  Server.Request.Charset := 'utf-8';


Alle Zeitangaben in WEZ +1. Es ist jetzt 17:36 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