Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Cross-Platform-Entwicklung (https://www.delphipraxis.net/91-cross-platform-entwicklung/)
-   -   Form laden: Localizeloading in InitHeritedcomponent (https://www.delphipraxis.net/196694-form-laden-localizeloading-initheritedcomponent.html)

Harry Stahl 10. Jun 2018 12:09


Form laden: Localizeloading in InitHeritedcomponent
 
Ich habe hier ein FMX-Project, wo das Laden der Mainform ca. 6-7 Sekunden dauert (ist aber erst seit einiger Zeit so).

Wenn ich per Debugging nach Application.Run durch den Code gehe, ist es der Aufruf in "InitInheritedComponent" (aus System.Classes) und zwar der Aufruf von der Procedure "NotifyGlobalLoading".

Ich verstehe dabei nicht so ganz, wie der Booleanwert für "LocalizeLoading" gesetzt wird oder ob man das irgendwie beeinflussen kann (so dass LocalizeLoading false ist und NotifiyGlobalLoading nicht aufgerufen werden muss).

Jemand eine Idee?

Delphi-Quellcode:
function InitInheritedComponent(Instance: TComponent; RootAncestor: TClass): Boolean;

  function InitComponent(ClassType: TClass): Boolean;
  begin
    Result := False;
    if (ClassType = TComponent) or (ClassType = RootAncestor) then Exit;
    Result := InitComponent(ClassType.ClassParent);
    Result := InternalReadComponentRes(ClassType.ClassName,
      FindResourceHInstance(FindClassHInstance(ClassType)), Instance) or Result;
  end;

var
  LocalizeLoading: Boolean;
begin
  GlobalNameSpace.BeginWrite; // hold lock across all ancestor loads (performance)
  try
    LocalizeLoading := (Instance.ComponentState * [csInline, csLoading]) = [];
    if LocalizeLoading then BeginGlobalLoading; // push new loadlist onto stack
    try
      Result := InitComponent(Instance.ClassType);
      if Result then Instance.ReadDeltaState;
      if LocalizeLoading then NotifyGlobalLoading; // < ----  call Loaded [das brauch Zeit!!]
    finally
      if LocalizeLoading then EndGlobalLoading; // pop loadlist off stack
    end;
  finally
    GlobalNameSpace.EndWrite;
  end;
end;
NotifyGloballoading sieht so aus:

Delphi-Quellcode:
procedure NotifyGlobalLoading;
var
  I: Integer;
  G: TList<TComponent>;
begin
  G := GlobalLoaded; // performance: eliminate repeated trips through TLS lookup
  for I := 0 to G.Count - 1 do
    TComponent(G[I]).Loaded;
end;

Uwe Raabe 10. Jun 2018 12:20

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Delphi-Quellcode:
LocalizeLoading := (Instance.ComponentState * [csInline, csLoading]) = [];

Bedeutet: LocalizeLoading wird auf True gesetzt, wenn die Schnittmenge zwischen Instance.ComponentState und [csInline, csLoading] leer ist. Mit anderen Worten, wenn Instance.ComponentState weder csInline noch csLoading enthält.

Einen Einfluss auf dieses Zustand hast du nicht, da diese Werte vom Streaming-System intern gesetzt werden.

Hast du irgendwo ein Loaded überschrieben?

Harry Stahl 10. Jun 2018 12:28

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Nein, habe da nichts überschrieben...

Uwe Raabe 10. Jun 2018 12:57

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Zitat:

Zitat von Harry Stahl (Beitrag 1404424)
Nein, habe da nichts überschrieben...

Kannst du irgendwie feststellen, wo die Zeit verbraucht wird?

Alternativ: Kannst du ein Testprojekt bereitstellen, das dieses Verhalten zeigt?

Harry Stahl 10. Jun 2018 14:15

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Ich habe mal eine lokale Kopie von der System.classes.pas gemacht und bin da mal durch die Procedure "NotifyGlobalLoading" mit verschiedenen Debugger-Einstellungen.

Ergebnis ist, dass es am MainMenu (laden der Komponente Menu1) liegt und zwar dann, wenn eine TImageList damit verbunden ist (ganze 6 Sekunden Unterschied).

Harry Stahl 10. Jun 2018 14:22

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Liste der Anhänge anzeigen (Anzahl: 1)
Habe mal eine leere Form erzeugt, da das Menu und die TImageList drauf, Project anbei.

Wenn man das compiliert und ausführt, dauert es einige Sekunden, bis sich die Form zeigt.

Wenn man den Link im Menü zur ImageList entfernt, neu kompiliert und ausführt, ist das Programm sofort da.

Wieso nur?

Uwe Raabe 10. Jun 2018 15:23

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Zitat:

Zitat von Harry Stahl (Beitrag 1404436)
Wieso nur?

Offenbar eine eher suboptimale Umsetzung des Windows-Menüs unter FMX.
Delphi-Quellcode:
TPlatformWin.UpdateMenuItem
interessiert sich nur mäßig für das aktuelle AItem (sucht immer das RootMenu) und gar nicht für AChange, den Grund des Update. Vielmehr löst jeder Aufruf dieser Routine einen Neuaufbau des gesamten Menüs aus. Da beim Setzen der ImageList aber jeder Menüpunkt durchlaufen wird, erfolgt für jeden Menüpunkt auch ein kompletter Neuaufbau.

Du kannst das einfach sehen, wenn du in deinem Beispiel einen Button mit folgendem Event zufügst:
Delphi-Quellcode:
  Menu1.Images := nil;
  Menu1.Images := imMain;
Ein Workaround ist ein bisschen tricky und auch unsauber. Deklariere vor deinem Form die folgende Interposer Class:
Delphi-Quellcode:
type
  TMenuItem = class(FMX.Menus.TMenuItem)
    procedure DoBitmapChanged; override;
  end;

type
  TForm16 = class(TForm)
    Menu1: TMainMenu;
...
procedure TMenuItem.DoBitmapChanged;
begin
  SetDesigning(True, False);
  inherited;
  SetDesigning(False, False);
end;
Das verhindert den jeweiligen Neuaufbau des Windows-Menüs. Daher musst du das selbst noch im FormCreate nachholen:
Delphi-Quellcode:
procedure TForm16.FormCreate(Sender: TObject);
begin
  Menu1.RecreateOSMenu;
end;
Kann gut sein, daß das für andere Plattformen nicht nötig ist oder dort sogar Probleme auslöst. Dann musst du den entsprechenden Code in passende IFDEFs packen.

Harry Stahl 10. Jun 2018 15:57

AW: Form laden: Localizeloading in InitHeritedcomponent
 
Super Uwe, ich bin beeindruckt.:thumb:

Ich quälte mich gerade auch durch die FMX.Menus.pas, um weiteres Licht ins Dunkel zu bringen.
Aber mit Deinem Work-Around kann ich leben, der funktioniert hier super. Musste in anderen Units, welche das Hauptformular includierten noch einige

tm: TMenuItem;

durch

tm: FMX.Menus.TMenuItem;

ersetzen, aber das war's dann auch schon.

Vielen Dank, das erleichtert mir auch das weitere Testen, auf einem anderen (langsameren) Rechner hat sich die Startzeit des Programms schon auf 20 Sekunden aufaddiert, jetzt geht's dort wieder in insgesamt 4 Sek (incl. laden einer Maildatei mit 50.000 Maileinträgen).

Übrigens: Bei TMenubar tritt das Problem nicht auf. TMainMenu hat aber wohl auch noch andere Probleme, z.B. dass das OnClickEvent eines Hauptmenüeintrages der Menüzeile nicht ausgeführt wird.


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