Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Dynamisches Package & Interface Typecast (https://www.delphipraxis.net/189158-dynamisches-package-interface-typecast.html)

Sequitar 10. Mai 2016 14:30

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;

Uwe Raabe 10. Mai 2016 14:45

AW: Dynamisches Package & Interface Typecast
 
Zitat:

"iExample" kennt die hauptanwendung ja nicht, da erst im package deklariert?
Hier liegt der Design-Fehler! Interfaces sind ja gerade dazu da, in verschiedenen Modulen deklariert zu sein. Das ist z.B. genau der Grund, warum COM-Server eine Type-Library zur Verfügung stellen: damit die Interfaces in Client bekannt sind.

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.

Sequitar 10. Mai 2016 14:59

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?

Uwe Raabe 10. Mai 2016 16:22

AW: Dynamisches Package & Interface Typecast
 
Zitat:

Zitat von Sequitar (Beitrag 1337893)
a) wieso ist das "with... as...do" nicht brauchbar? ich dachte, das sei besser als hardcast (
Delphi-Quellcode:
iexample(i)
)

Das
Delphi-Quellcode:
as
ist in Ordnung aber
Delphi-Quellcode:
with
nicht: http://hallvards.blogspot.de/2004/08...d-harmful.html

Zitat:

Zitat von Sequitar (Beitrag 1337893)
b)Einbindung der Interfaces in die Hauptanwendung

Dann solltest du die Interfaces deklarieren, die alle Methoden enthalten, mit denen die Hauptanwendung etwas anfangen kann. Ob du dann diese Interfaces in den Packages ableitest oder separate deklarierst, ist nebensächlich. Über Supports kannst du überprüfen, ob die Klasse das jeweilige Interface implementiert und dir gleichzeitig auch eine Instanz dieses Interfaces zurückgeben lassen.

Delphi-Quellcode:
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;
Im Package kannst du dann z.B. über abgeleitete Interfaces die speziellen Dinge realisieren.

Delphi-Quellcode:
type
  ITestExt = interface(ITest)
    procedure TestFall1;
  end;

  IMachWasAnderes = interface(IMachWas)
    procedure DasSollKeinerWissen;
  end;
Der
Delphi-Quellcode:
AS
-Cast auf
Delphi-Quellcode:
TInterfacedPersistent
ist auch überflüssig, da
Delphi-Quellcode:
Supports
bereits für
Delphi-Quellcode:
TObject
funktioniert.

Zitat:

Zitat von Sequitar (Beitrag 1337893)
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?

Nein, nicht ohne erheblichen und völlig unnötigen Aufwand.

Zitat:

Zitat von Sequitar (Beitrag 1337893)
oder ist das ein genereller design fehler des systems?

Offensichtlich.

Zitat:

Zitat von Sequitar (Beitrag 1337893)
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).

Method Pointer sind hier völlig fehl am Platze. Du machst es dir damit eindeutig zu schwer. Compiliere die Interface-Deklarationen (sinnvollerweise in einer eigenen Unit von den implementierenden Klassen getrennt) in die Haup0tanwendung ein und die Sache ist geritzt.

Zitat:

Zitat von Sequitar (Beitrag 1337893)
Kann ich dann auch auf Properties zugreifen?

Wenn das Interface diese deklariert, sicher.

Sequitar 10. Mai 2016 16:40

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.

Sequitar 20. Mai 2016 19:33

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.

Blup 31. Mai 2016 15:56

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:

http://www.delphipraxis.net/186326-i...ufrufen-2.html


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