![]() |
Delphi-Version: 10 Seattle
Dynamisches Package & Interface Typecast
Hallo, ich versuche gerade auf methoden in dynamisch geladenen packages zuzugreifen.
Zur Vereinheitlichung der enthaltenen Klassen wird hierzu ein basis interface angegeben (gemeinsames, statisch gelinktes package, welches dann die anderen nachlädt). Da die ladende EXE also den content der dynamisch geladenen packages nicht kennt, werden die objecte nach einsatz von "getclass" erst als tinterfacedpersistent gecastet und dann auf den interface support getestet. Falls der test positiv, dann sollten die im interface delarierten methoden aufgerufen werden. Nun also zum Problem:
Delphi-Quellcode:
Function Tplgloader.Getinterfacedclass(Classname: String; Intf: Tguid) : Tpersistentclass; Begin Result := Getclass(Classname); Assert(Assigned(Result), 'Class <' + Classname + '> not found.'); Assert(Supports(Result, Intf), 'Class <' + Classname + '> does not support provided interface.') End; //hauptanwendung begin With Loader Do // Globaler loader ist vom typ Tplgloader, lädt,entlädt und verwaltet packages Try Begin //guid of interface iexample Guid := Tguid.Create('{EB84A8B3-0A16-4746-9034-47E81990F6D9}'); //wurde dem interface "iexample" im package zugeordnet I := Getinterfacedclass('texampleclass', Guid).Create As Tinterfacedpersistent; If Supports(I, Guid) //ok Then //>> Problemstelle: typecast und methodenaufruf With I As iexample Do //wie kann man zum aufruf auf das interface typecasten, "iExample" kennt die hauptanwendung ja nicht, da erst im package deklariert? Begin //und somit die methoden des interfaces aufrufen?? Init('aaa', True); // zur Zeit nicht möglich End; I.Free; End; Finally End; End; |
AW: Dynamisches Package & Interface Typecast
Zitat:
Die GUID von iExample musst du ja sowieso kennen, und wenn du was davon aufrufen willst, musst du auch die Methodennamen kennen. Also mach dir das Leben einfach und binde die Interface-Deklaration in die Hauptanwendung ein. OT: Bitte kein WITH verwenden! Jedenfalls nicht in diesem Kontext. |
AW: Dynamisches Package & Interface Typecast
Erst mal Danke für die Rückmeldung.
Dazu zwei Fragen a) wieso ist das "with... as...do" nicht brauchbar? ich dachte, das sei besser als hardcast (
Delphi-Quellcode:
)
iexample(i)
b)Einbindung der Interfaces in die Hauptanwendung Nehmen wir mal an ich möchte so unterschiedlichste klassen / funktionalitäten dynamisch nachladen, die aber eigentlich nichts miteinander zu tun haben. Also haben sie auch fast keine gemeinsamkeiten für ein einziges interface in der hauptanwendung (sonst wäre das natürlich der logischste weg), sondern nur innerhalb der packages (zb ein datenbank interface /-paket, ein sprach-interface /-paket etc) aber was ist wenn ich zum beispiel im package 1
Delphi-Quellcode:
type itest=interface
... end; ttest1=class (tinterfacedobject, itest) ... end ttest2=class(tinterfacedobject, itest) ... end; aber in einem anderen package andere funktionalität erwünscht ist und somit
Delphi-Quellcode:
type imachwas=interface
... end; tmachwas1=class (tinterfacedobject, imachwas) ... end tmachwas2=class(tinterfacedobject, imachwas) ... end; habe ich da keine möglichkeit, quasi die funktionalitäten aus den einzelenen packages anhand der interfaces zu "erkennen", ohne dass beide unbedingt in der hauptanwendung deklariert sind? oder ist das ein genereller design fehler des systems? Edit: Was Sinn machte wäre zb ein gemeinsames Interface, um der hauptanwendung informationen zu den einzelen Packages zu liefern. Nur wie kann ich dann am einfachsten auf die restlichen inhalte zugreifen (dynamisches laden und aufrufen von methoden mithilfe von method pointers ist mir bekannt). Die Frage ist dann WIE kann ich der hauptanwendung mitteilen, welche parameter die methodenNAMEN in den packages verwenden? Eine liste verfügbarer methoden kann ich bereits erstellen. Kann ich dann auch auf Properties zugreifen? |
AW: Dynamisches Package & Interface Typecast
Zitat:
Delphi-Quellcode:
ist in Ordnung aber
as
Delphi-Quellcode:
nicht:
with
![]() Zitat:
Delphi-Quellcode:
Im Package kannst du dann z.B. über abgeleitete Interfaces die speziellen Dinge realisieren.
type itest=interface
function GetDescription: string; procedure DoTest; end; type imachwas=interface function GetName: string; procedure Execute; end; var myTestIntf: ITest; myMachWasIntf: IMachWas; begin if Supports(myClass, ITest, myTestIntf) then begin myTestIntf.DoTest; end; if Supports(myClass, IMachWas, myMachWasIntf) then begin myMachWasIntf.Execute; end; end;
Delphi-Quellcode:
Der
type
ITestExt = interface(ITest) procedure TestFall1; end; IMachWasAnderes = interface(IMachWas) procedure DasSollKeinerWissen; end;
Delphi-Quellcode:
-Cast auf
AS
Delphi-Quellcode:
ist auch überflüssig, da
TInterfacedPersistent
Delphi-Quellcode:
bereits für
Supports
Delphi-Quellcode:
funktioniert.
TObject
Zitat:
Zitat:
Zitat:
Zitat:
|
AW: Dynamisches Package & Interface Typecast
Super, merci für diese ausführliche Hilfestellung / Anleitung. Hat den Blick v.a. auf Fallstricke und Fehler in der bisherigen Herangehensweise gelenkt.
Ich werde jetzt erst mal versuchen, das bestehende System auf die Vorschläge hin abzuändern. |
AW: Dynamisches Package & Interface Typecast
Liste der Anhänge anzeigen (Anzahl: 1)
Habe mir jetzt mal ein einfaches Testprogramm geschrieben. Kann jetzt also das package dynamisch laden und enstprechende interfacedobjects anhand ihrer Klassennamen erstellen und auf den Interface-Support testen.
Zu Zwei Dinge... a) Ich kann bisher nicht den korrekten Destructor aufrufen (müsste das nicht eigentlich automatisch passieren, sobald man einem interface:=nil zuordnet?) und somit die Objekte nur anhand ihrer gemeinsamen Basisklasse freigeben. Anbei ein kleines Beispielprogramm zur Veranschaulichung b) das Entladen der Library crasht mit einer access violation in der RTL, sobald ich mehr als ein interfaced objekt daraus erstelle. Dies betrifft bisher nur das angefügte Testprogramm. In Anderen Programmen kann ich bisher beliebige packages laden, nutzen und entladen. |
AW: Dynamisches Package & Interface Typecast
Der Destructor Destroy von TInterfacedObject ist virtuell und wird automatisch aufgerufen, sobald der Referenzzähler auf Null fällt.
Alle Nachkommen müssen diesen Destruktor überschreiben, wenn zusätzliches Verhalten bei der Freigabe erforderlich ist. Ich halte deinen Entwurf für ein Plugin-System für viel zu aufwendig und schlecht erweiterbar. Wir hatten so ein Thema schon mal: ![]() |
Alle Zeitangaben in WEZ +1. Es ist jetzt 16:44 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