![]() |
Delphi-Version: 5
TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
Mal wieder so ne Frage, bei der ihr euch wahrscheinlich denkt: "Mensch, kann der überhaupt irgendwas?", but here we go :lol:
Ich habe eine HashMap (also TDictionary), die als Values Listen mit Integers (TList<Integer>) enthält. Gefüllt wird die HashMap über eine Schleife. Aber ich bekomme es nicht hin, dass die Values der HashMap einzigarte Werte sind. Im Objektinspektor steht da bei allen Einträgen immer die selbe Speicheradresse. In Java würde ich mir bei sowas halt in jedem Schleifendurchlauf ein neues Listenobjekt erzeugen und dieses dann (entsprechend gefüllt) der HashMap zufügen. Aber in Delphi kann man ja keine Variablen im Anweisungsblock deklarieren, daher stehe ich diesbezüglich etwas aufm Schlauch... :-( |
AW: Case Syntax
Ich fürchte, ich habe kaum die Hälfte verstanden. Zeig doch mal etwas Code, dann wird das Problem vielleicht klarer.
|
AW: Case Syntax
Delphi-Quellcode:
Grüße
repeat
objVar := TMyObject.create; //hier wird die vorherige Instanz überschrieben // do something with instance Hashmap.addItem(objVar); until SomeCondition Klaus |
AW: Case Syntax
So?
Delphi-Quellcode:
uses
Generics.Collections; type TMultiMap<TKey,TValue> = class private fHashMap: TDictionary<TKey,TList<TValue>>; public constructor Create; destructor Destroy; override; procedure Add(const key: TKey; const value: TValue); end; constructor TMultiMap<TKey, TValue>.Create; begin inherited Create; fHashMap := TObjectDictionary<TKey,TList<TValue>>.Create([doOwnsValues]); end; destructor TMultiMap<TKey, TValue>.Destroy; begin fHashMap.Free; inherited; end; procedure TMultiMap<TKey, TValue>.Add(const key: TKey; const value: TValue); var list: TList<TValue>; begin if not fHashMap.TryGetValue(key, list) then begin list := TList<TValue>.Create; fHashMap.Add(key, list); end; list.Add(value); end; |
AW: TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
Zitat:
Und ich zudem in der Hast das Wort "einzigartig" unbedacht verwendet habe - gemeint ist: im Speicher distinkte Objekte Klaus: das habe ich probiert, ohne Erfolg jedoch. Wahrscheinlich liegt das daran, dass zwar ein neues Objekt erzeugt wird, durchaus aber mit der gleichen Adresse im Speicher. Und wenn im Dictionary nur Adressen als Values abgelegt sind, zeigen die halt dann immer noch alle auf das gleiche Objekt. Stevie: danke für die Mühe, vor allem, weil du auch beantwortet hast, wonach ich streng genommen gefragt hab... Ich sollte diese Fragen nicht in einer Minute vor nem Meeting hinklatschen. Weil die Frage nach Code kam:
Delphi-Quellcode:
Angenommen, ich möchte die natürlichen Zahlen in Intervalle einteilen, die sich durch die Fibonacci-Zahlen definiert sind: Intervall i := [fib(i-1)+1,fib(i)].
var
vHashMap : TDictionary<Integer,TList<Integer>>; vList : TList<Integer>; state : Integer; begin vHashMap := TDictionary<Integer,TList<Integer>>.Create; vList := TList<Integer>.Create; while condition do case state of 0: begin vList.Add(anInteger); state := 1; end; 1: begin if something do vList.Add(anInteger); else begin vHashMap.Add(aKey,vList); vList.Clear; state := 0; end; end; else [...] end; end; Die HashMap soll dann die einzelnen Intervalle enthalten, der Schlüssel sei i. (Anmerkung: dies ist nicht der eigentliche Sinn des Codes. Es ist lediglich ein greifbares und passendes Beispiel) Lassen wir i von 3 bis 6 laufen (3, weil die geforderten Intervalle für i < 3 nicht möglich sind). Die HashMap sollte nach der Schleife dann folgendermaßen aussehen: ((3,{2}),(4,{3}),(5,{4,5}),(6,{6,7,8})) So, wie ich es im Code-Ausschnitt implementiert habe, würde es allerdings folgendermaßen aussehen: ((3,{6,7,8}),(4,{6,7,8}),(5,{6,7,8}),(6,{6,7,8})) Der Grund dafür ist auch klar: das Dictionary-Objekt speichert nicht selbst den Wert eines Eintrages, sondern nur dessen Speicheradresse. Und die ist für vList natürlich immer gleich. Die Frage ist also: wie macht man das richtig in Delphi? In Java würde ich mir in solchen Fällen halt einfach ein neues Objekt von TList erzeugen, wenn ichs brauche, und dieses der HashMap übergeben. Solange noch ne Referenz auf das Listen-Objekt besteht, was ja der Fall ist, wenn es in der Map drin steht, bleibt es auch im Speicher. |
AW: TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
Warum so kompliziert?
Delphi-Quellcode:
Gut, das kann man nun noch optimieren, da man weiß, dass fib(i) eh auch fib(i-1) berechnet, aber nunja
var
map: TMultiMap<Integer,Integer>; i, f: Integer; begin map := TMultiMap<Integer,Integer>.Create; for i := 3 to 6 do for f := fib(i-1)+1 to fib(i) do map.Add(i, f); |
AW: TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
Zitat:
Weil mein eigentlicher Code nichts mit Fibonacci-Zahlen zu tun hat :-D |
AW: TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
Zitat:
Würde man in Java übrigens auch so machen und die Multimap aus guava oder sonstwoher nutzen, damit man sich den Krams spart. |
AW: TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
würde es auch so funktionieren?
Delphi-Quellcode:
Grüße
begin
vHashMap.Add(aKey,vList); vList.free; vList := TList<Integer>.Create; state := 0; end; Klaus |
AW: TDictionary - mit einem Bezeichner unterschiedliche Values einfügen
Zitat:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 12:13 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