![]() |
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:
NotifyGloballoading sieht so aus:
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;
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; |
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? |
AW: Form laden: Localizeloading in InitHeritedcomponent
Nein, habe da nichts überschrieben...
|
AW: Form laden: Localizeloading in InitHeritedcomponent
Zitat:
Alternativ: Kannst du ein Testprojekt bereitstellen, das dieses Verhalten zeigt? |
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). |
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? |
AW: Form laden: Localizeloading in InitHeritedcomponent
Zitat:
Delphi-Quellcode:
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.
TPlatformWin.UpdateMenuItem
Du kannst das einfach sehen, wenn du in deinem Beispiel einen Button mit folgendem Event zufügst:
Delphi-Quellcode:
Ein Workaround ist ein bisschen tricky und auch unsauber. Deklariere vor deinem Form die folgende Interposer Class:
Menu1.Images := nil;
Menu1.Images := imMain;
Delphi-Quellcode:
Das verhindert den jeweiligen Neuaufbau des Windows-Menüs. Daher musst du das selbst noch im FormCreate nachholen:
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;
Delphi-Quellcode:
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.
procedure TForm16.FormCreate(Sender: TObject);
begin Menu1.RecreateOSMenu; end; |
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 18:30 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