![]() |
Generics Verständnisproblem
Hallo!
Ich arbeite gerade das erste Mal mit Generics in Delphi 2009 und habe ein Verständnisproblem. Ich möchte eine Funktion mit zwei verschiedenen Datentypen verwenden und mir die Redundanz ersparen, den Code an eine andere Stelle zu kopieren oder mit Verzweigungen zu arbeiten. Die Funktion sieht in etwa so aus:
Delphi-Quellcode:
Die zweite Funktion soll 1:1 das gleiche machen, nur ist Data dort vom Typ PListsDataRecord, statt PFrequencyItem.
procedure PerformPostAnalysis(List: TBaseVirtualTree; Threshold: Integer);
var Node: PVirtualNode; Data: PFrequencyItem; AnalysisScore, MaxScoreIdx: Double; begin try Node := List.GetFirst; while Assigned(Node) do begin Data := List.GetNodeData(Node); // MaxScoreIdx ermitteln // ... AnalysisScore := Data^.AnalysisScore + (1 - Data^.AnalysisScore) * (MaxScoreIdx / 10) if AnalysisScore < (Threshold / 100) then begin // ... end; Node := List.GetNext(Node); end; except // ... end; end; Also habe ich folgendes versucht:
Delphi-Quellcode:
Dabei tauchen zwei Probleme auf:
type
TFrequencyPostAnalysis<T> = class class procedure Perform(List: TBaseVirtualTree; Threshold: Integer); end; // ... class procedure TFrequencyPostAnalysis<T>.Perform(List: TBaseVirtualTree; Threshold: Integer); var Node: PVirtualNode; Data: T; AnalysisScore, MaxScoreIdx: Double; begin try Node := List.GetFirst; while Assigned(Node) do begin Data := List.GetNodeData(Node); // MaxScoreIdx ermitteln // ... AnalysisScore := Data^.AnalysisScore + (1 - Data^.AnalysisScore) * (MaxScoreIdx / 10) if AnalysisScore < (Threshold / 100) then begin // ... end; Node := List.GetNext(Node); end; except // ... end; end; 1) List.GetNodeData liefert einen Pointer zurück. Da Data aber vom Typ T ist, meckert der Compiler: Inkompatible Typen: 'T' und 'Pointer'. Also habe ich eine Variable P: Pointer angelegt, P := List.GetNodeData; Data := T(P^). Das compiled er auch, meckert aber sofort an der nächsten Stelle: 2) Wenn Data vom Typ T ist, kennt Delphi Data.AnalysisScore (logischerweise) nicht mehr. Wie kann ich dem Compiler sagen, was für Eigenschaften Data hat? Ich kann natürlich der Deklaration der Generics-Klasse sagen, dass sie TFrequencyPostAnalysis<TListsDataRecord> oder TFrequencyPostAnalysis<TFrequencyItem> sein soll. Aber das macht ja dann wiederum keinen Sinn, wenn es darum geht, mit dem anderen Typen zu arbeiten ...?! :wall: Wie kann ich diesen Knoten also aufdröseln? Funktioniert das, was ich vorhabe, überhaupt mit Generics? |
AW: Generics Verständnisproblem
Ich glaube mit Generics wird das schwierig - zumindest so wie Du es vor hast.
Du bräuchtest so was wie einen abstrakten Record, von dem deine beiden anderen Record-typen ableiten und Dir somit über einen gemeinsamen Vorfahren "AnalysisScore" zur Verfügung stellen könnten -- Vererbung geht aber mit Records halt nicht. Und da der VirtualTreeView nun mal damit arbeitest, wird es so nicht gehen. Wenn Du aber nur diese eine "AnalysisScore"-Eigenschaft brauchst, könntest Du es über einen Funktionszeiger lösen: Du definiert eine Funktion die als Eingabe einen PVirtualNode bekommt und einen Double (Den AnalysisScore) zurück gibt. Diese Funktion übergibst Du dann als Parameter an die "Perform"-Funktion. Für jeden TreeView-Typ kannst Du also - abhängig von dessen NodeData record - so eine Funktion schreiben bzw. dann übergeben... Viele Grüße |
AW: Generics Verständnisproblem
Delphi-Quellcode:
und T=TListsDataRecord könnte gehn
Data: ^T;
PS: Bei sowas wie Data^.AnalysisScore könnte man das ^ weglassen, da Delphi bei einem Zeigertypen, gefolgt von einem Punkt, implizit auch eine Dereferenzierung vornehmen würde. Macht sich z.B. gut, wenn man doch irgendwann mal von einem Record auf ein Datenobjekt umsteigen möchte. |
AW: Generics Verständnisproblem
Zitat:
Delphi-Quellcode:
Oder:
var
CurrentNodeData: PMyClass; begin CurrentNodeData := PMyClass(Sender.GetNodeData(Node));
Delphi-Quellcode:
So ca. mache ich das immer.
var
CurrentNodeData: TMyClass; begin CurrentNodeData := TMyClass(Sender.GetNodeData(Node)^); |
AW: Generics Verständnisproblem
Oh ja, stimmt, Du hast Recht, daran hatte ich gar nicht gedacht :). Wenn man es so macht, geht es wie oben genannt mit den Generics. :) Einfach einen gemeinsamen Vorfahren definieren, der AnalysisScore bereitstellt (oder halt ein Interface)...
|
AW: Generics Verständnisproblem
Zitat:
Grüße |
AW: Generics Verständnisproblem
Im Falle von Objekten geht dann auch
Delphi-Quellcode:
,
type TFrequencyPostAnalysis<T: object> = class
womit man einmal sicherstellt, daß der generische Typ immer ein Objekt ist und man hat dann auch auf die Grundfunktionen aller Objekte zugriff, wie z.B.
Delphi-Quellcode:
und
Data.Free;
Delphi-Quellcode:
.
Data is TMyClass
Praktischer wäre es ja, wen VTV mal mit Generics aufgemotzt würde. :stupid: |
AW: Generics Verständnisproblem
Zitat:
|
AW: Generics Verständnisproblem
Danke für eure Tipps! :)
Zitat:
Zitat:
Zitat:
Edit: Zum besseren Verständnis, PListsDataRecord und PFrequencyItem werden ja durch den ganzen restlichen Code hinweg für die Datenverwaltung der jeweiligen Bäume genutzt und enthalten verschiedene unterschiedliche Felder. Wenn ich also zB eine PMyClass anlege, die nur die für die Funktion gemeinsam benötigten Felder beinhaltet, liefert der VTV bei GetNodeData ja "Müll", wenn die Struktur des Zielrecords (und damit auch die SizeOf) von der NodeDataSize des jeweiligen Trees abweicht. Oder soll dann PMyClass die "alten" Records ersetzen? |
AW: Generics Verständnisproblem
Zitat:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 06:15 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 by Thomas Breitkreuz