Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Algorithmen, Datenstrukturen und Klassendesign (https://www.delphipraxis.net/78-algorithmen-datenstrukturen-und-klassendesign/)
-   -   RTTI Problem (https://www.delphipraxis.net/215010-rtti-problem.html)

TurboMagic 21. Apr 2024 16:18

RTTI Problem
 
Hallo,

ich versuche gerade mein Glück mit Rtti und habe mir mit Hilfe der Hilfe
diesen Code zusammengestoppelt, der jedoch leider nicht funktioniert:

Delphi-Quellcode:
procedure TProduct.Assign(Product: IProduct);
var
  LContext     : TRttiContext;
  LInstanceType : TRttiInstanceType;
  LType        : TRttiType;
  LInterface   : TRttiInterfaceType;
  Props        : TArray<TRttiProperty>;
  Value        : TValue;
begin
  LContext := TRttiContext.Create;
  try
    LType        := LContext.GetType(TProduct);
    LInstanceType := LType.AsInstance;
    LInstanceType.GetInterface(TGUID.Create('{108A3B28-575B-423F-AE2E-1D8354E3289C}'),
                               LInterface);
    Props        := LInterface.GetProperties;
end;
Was geht nicht?
LInterface ist nil, also scheitert GetInterface.
Das IProduct Interface besitzt jedoch die im Code angegebene GUIID und wenn ich da rein debugge
wird diese GUIID scheinbar auch richtig erzeugt. Nur das Interface wird intern irgendwie nicht
gefunden.

Nein, RTTI Hab' ich in der betreffenden Unit nicht ausgemacht. IProduct erbt von einem anderen
selbst definierten Interface, das hat aber seine eigene GUID und erbt nicht weiter (bzw. IInterface).

Woran liegt das, dass das nicht gefunden wird?
Das Programm benutzt zur Laufzeit schließlich erfolgreich jede Menge IProducts...

Uwe Raabe 21. Apr 2024 16:31

AW: RTTI Problem
 
Der Aufruf LInstanceType.GetInterface versucht das gefragte Interface von LInstanceType (also einer Instanz von TRttiInstanceType) zu bekommen. GetInterface ist in TObject deklariert und hat mit RTTI überhaupt nichts zu tun.

Ich bin mir noch nicht ganz sicher was du erreichen willst, daher kann ich nicht sagen wie es richtig wäre.

TurboMagic 21. Apr 2024 16:53

AW: RTTI Problem
 
Ok, was ich erreichen will ist eigentlich einfach:

- ermittle eine Liste aller Properties des IProduct Interfaces
- iteriere durch die Liste
- lies den jeweiligen property Wert des Product: IProduct parameters
- weise diesen dem selben Property der self Instanz zu (die ist ja auch ein TProduct und somit ein IProduct)

Uwe Raabe 21. Apr 2024 17:39

AW: RTTI Problem
 
Zitat:

Zitat von TurboMagic (Beitrag 1536002)
- ermittle eine Liste aller Properties des IProduct Interfaces

Damit wirst du wohl kein Glück haben: RSP-21395

TurboMagic 21. Apr 2024 17:52

AW: RTTI Problem
 
Ok, wen das so ist, muss ich es halt "zu Fuß" umsetzen.
Geht auch, man muss halt nur bei jeder Erweiterung dran denken das nachzuziehen...

QuickAndDirty 22. Apr 2024 12:17

AW: RTTI Problem
 
Edit:
War quatsch

Uwe Raabe 22. Apr 2024 12:44

AW: RTTI Problem
 
Zitat:

Zitat von TurboMagic (Beitrag 1536006)
muss ich es halt "zu Fuß" umsetzen.

Oft ist das auch die saubere Herangehensweise, denn nicht immer sollen oder können alle Properties so einfach kopiert werden. Auch dabei sind dann die zukünftigen Erweiterungen die Stolperfallen, die man meist nicht auf dem Schirm hat (weil, "läuft ja alles automatisch").

himitsu 22. Apr 2024 13:04

AW: RTTI Problem
 
Nja, interfaces besitzen offiziell ja auch keine Property, auch wenn mehrere Compiler sowas anbieten (teilweise auch automatisch, für GetXXX- und SetXXX-Methoden).
OK, in der Interfacedeklaration kann es somit nicht drin sein, aber hätte eigentlich dennoch erwartet, dass es Delpgi dennoch in der RTTI auflistet. :shock:


Bei Records, welche eventuell irgendwann mal erweitert werden, füge ich manchmal an solchen Stellen ein
Delphi-Quellcode:
{$IF SizeOf(TMyRecord) <> 123} {MESSAGE Warn 'guck ma'} {$IFEND}
(oder Hint/Error) ein.

Ist das ein eigenes Interface?
Hier wäre es dann vielleicht einfacher, wenn du dir einen anderen Standard ansiehts. (Delphi-Referenz durchsuchenTPersistent.Assign)
Keine Ahnung, ob man für Interfaces auch einen Record-Helper schreiben könnte, falls es nichts Eigenes ist.

Dennis07 23. Apr 2024 09:25

AW: RTTI Problem
 
Zitat:

Zitat von himitsu (Beitrag 1536023)
OK, in der Interfacedeklaration kann es somit nicht drin sein, aber hätte eigentlich dennoch erwartet, dass es Delpgi dennoch in der RTTI auflistet. :shock:

Interface-Properties sind, wie du oben schriebst, halt keine "echten" Properties. Sie haben auch noch andere Einschränkungen. Sie sind wirklich ausschließlich "semantischer Zucker" und werden vom Compiler nicht weiter beachtet. Interfaces-Properties haben deshalb keine RTTI, du musst sie also entweder über die Klassentypen abgreifen (ginge ja auch), oder über den Getter/Setter-Name.

Zitat:

Zitat von himitsu (Beitrag 1536023)
Keine Ahnung, ob man für Interfaces auch einen Record-Helper schreiben könnte, falls es nichts Eigenes ist.

Kann man nicht.

jaenicke 23. Apr 2024 10:20

AW: RTTI Problem
 
Für den Zweck könnte man auch einen Pascal-Skriptparser verwenden. Wenn man die Units mit den Interfaces im Buildprozess als Ressource anhängt, kann man die Informationen von dort ziehen. Da die gleichen Dateien auch zum Kompilieren verwendet werden, sind die Daten auch immer aktuell.

Uwe Raabe 23. Apr 2024 11:17

AW: RTTI Problem
 
Wie ich schon schrieb halte ich den Ansatz, die über die Properties zu iterieren und diese jeweils zu kopieren, - wenn überhaupt mit vertretbarem Aufwand umsetzbar - für ziemlich aufwändig, fehleranfällig und schlecht wartbar. Immerhin muss bei jedem neuen Property geprüft werden, ob sich das so einfach kopieren lässt und das auch sachlich korrekt durchgeführt wird. Nicht immer enthalten die Properties Strings oder numerische Werte. Klasseninstanzen, Interfaces oder auch Arrays werden so nur als Referenzen und nicht inhaltlich kopiert. Das ist in der Regel bei Arrays und oft bei Klassen gar nicht gewollt. Solche Fälle müssten jeweils gesondert behandelt werden, was bei der RTTI-Lösung den Aufwand beträchtlich steigert.

Den empfohlenen Ansatz hat Frank schon mit TPersistent.Assign und seinem Pendant TPersistent.AssignTo angesprochen. Dieses Verfahren ist schon so alt wie Delphi, wird allgemein verstanden und hat sich bei korrekter Anwendung als stabil erwiesen.

Man kann sich die Implementierung in den simplen Fällen (Strings und numerische Werte) vereinfachen, wenn man die einzelnen Felder hinter den Properties einen record verlagert, der mit einfacher Zuweisung kopiert werden kann. Das deckt die ursprüngliche Absicht zur Vereinfachung der Wartbarkeit bereits ab. Da in dem beschriebenen Fall offenbar mit Interfaces gearbeitet wird, sind sowieso schon Getter und Setter vorhanden, die dann auf die Record-Felder umgeleitet werden. Eine mögliche Implementierung der TPersistent.Assign Ableitung sähe dann, unter der Annahme dass TProduct direkt von TPersistent abgeleitet ist, in etwa so aus:
Delphi-Quellcode:
procedure TProduct.Assign(Source: TPersistent);
begin
  if Source is TProduct then begin
    FData := TProduct(Source).FData;
  end
  else
    inherited;
end;
Bei dem Assign mit einem IProduct als Source wird es etwas komplexer und es bieten sich je nach Gegebenheiten mehrere Lösungsmöglichkeiten an:
  • Man erweitert IProduct um eine entsprechende Methode, die dann in allen Implementierungen entsprechend gefüllt werden muss.
  • Man deklariert ein separates Interface für das Assign, das man dan über Supports abfragt.
  • Wenn sichergestellt ist, dass alle Implementationen von IProduct auf TPersistent aufbauen:
    Delphi-Quellcode:
    Assign(Source as TPersistent);

himitsu 23. Apr 2024 11:18

AW: RTTI Problem
 
"beachten" muß der Compiler diese Interface-Property schon,
* denn man kann sie verwenden (wie bei normalen Property wird beim Auslesen/Zuweisen im Code einfach direkt gegen die Getter und Setter kompiliert)
* in den DCU müssen Infos für diese Property ebenfalls irgendwie enthalten sein, damit der PropertyName dann aufgelöst werden kann
* in BPL müssten sie auch drin stehen, damit nach Änderung auch der richtige Getter/Setter aufgerufen wird
:gruebel:

OK, unabhängig davon seht es in der TypeInfo des Interfaces definitiv nicht drin und scheinbar leider auch nicht in der Delphi-TypeInfo und RTTI.




Wenn es sich um Interfaces vom Delphi handelt, also mit einem TObjekt dahinter,
dann könnte man das Interface nach TObject casten und stattdessen über die RTTI des Objekts gehn.


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