![]() |
Delphi-Version: XE2
Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Ich habe hier mal ein ganz merkwürdenes Verhalten bei einer generischen sortierten Liste.
Die Einträge werden sehr hübsch sortiert ... aber fragt man jetzt nach dem Index von einem Element, dann kommt die Liste wohl etwas durcheinander. Hier mal die Ausgabe von dem Programm:
Code:
Irgendwie seltsam ... hätte ich auf jeden Fall nicht so erwartet, zumal die Sortierung auch manuell angestossen werden muss ... fügt man weitere Elemente hinzu, so ist die Liste nicht automatisch sortiert, aber für die Bestimmung des Item-Index wird der Comparer bemüht ...
ID Value List.IndexOf
== ===== ============ 20 4080 0 12 4081 1 11 4081 1 19 4081 1 13 4081 1 1 4081 1 10 4082 6 3 4272 7 2 4272 7 16 4272 7 5 4272 7 7 4272 7 21 4272 7 4 4273 13 9 4304 14 17 4313 15 8 4313 15 6 4313 15 14 4313 15 15 4314 19 18 4314 19 22 38892 21 Hat jemand dafür eine Erklärung und ist das ein gewolltes Verhalten? Hier der Source:
Delphi-Quellcode:
program SortList;
{$APPTYPE CONSOLE} {$R *.res} uses System.SysUtils, System.Generics.Collections, System.Generics.Defaults; type TItem = class private FValue : Integer; FID : Integer; public constructor Create( AID, AValue : Integer ); property ID : Integer read FID write FID; property Value : Integer read FValue write FValue; end; { TItem } constructor TItem.Create( AID, AValue : Integer ); begin inherited Create; FID := AID; FValue := AValue; end; procedure TestRun; var LList : TList<TItem>; LItem : TItem; begin LList := TObjectList<TItem>.Create( TComparer<TItem>.Construct( function( const L, R : TItem ) : Integer begin Result := L.Value - R.Value; end ) ); LList.Add( TItem.Create( 1, 4081 ) ); LList.Add( TItem.Create( 2, 4272 ) ); LList.Add( TItem.Create( 3, 4272 ) ); LList.Add( TItem.Create( 4, 4273 ) ); LList.Add( TItem.Create( 5, 4272 ) ); LList.Add( TItem.Create( 6, 4313 ) ); LList.Add( TItem.Create( 7, 4272 ) ); LList.Add( TItem.Create( 8, 4313 ) ); LList.Add( TItem.Create( 9, 4304 ) ); LList.Add( TItem.Create( 10, 4082 ) ); LList.Add( TItem.Create( 11, 4081 ) ); LList.Add( TItem.Create( 12, 4081 ) ); LList.Add( TItem.Create( 13, 4081 ) ); LList.Add( TItem.Create( 14, 4313 ) ); LList.Add( TItem.Create( 15, 4314 ) ); LList.Add( TItem.Create( 16, 4272 ) ); LList.Add( TItem.Create( 17, 4313 ) ); LList.Add( TItem.Create( 18, 4314 ) ); LList.Add( TItem.Create( 19, 4081 ) ); LList.Add( TItem.Create( 20, 4080 ) ); LList.Add( TItem.Create( 21, 4272 ) ); LList.Add( TItem.Create( 22, 38892 ) ); LList.Sort; WriteLn( 'ID':4, 'Value':7, 'List.IndexOf':14 ); WriteLn( '==':4, '=====':7, '============':14 ); for LItem in LList do begin WriteLn( LItem.ID : 4, LItem.Value : 7, LList.IndexOf( LItem ) : 14 ); end; end; begin try TestRun; except on E : Exception do WriteLn( E.ClassName, ': ', E.Message ); end; Readln; end. |
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Wenn man sich die Implementierung von
Delphi-Quellcode:
anschaut ....
function TList<T>.IndexOf(const Value: T): Integer;
Gegf. könntest Du
Delphi-Quellcode:
erweitern um einen Vergleich von ID wenn Value identisch ist.....
function( const L, R : TItem ) : Integer
begin Result := L.Value - R.Value; end ) |
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Sieht aus als ob IndexOf nicht nach dem Pointer(Objektinstanz) in der Liste schaut sondern per Comparer nach Übereinstimmung sucht und dann den ersten Index zurück gibt bei dem die Werte übereinstimmen.
|
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Ich habe den Quellcode nicht, aber falls IndexOf iterativ umgesetzt ist, ist das Verhalten doch vollkommen ok.
Deine Ordnungsfunktion ist nicht vollständig implementiert, d.h. es gilt eben nicht: A != B => f(A) != f(B) Lösung? => Bummi Zitat:
Die Liste ist ein Container für Elemente. Mit der Definition einer Ordnungsrelation ('Comparer') und der Sort-Methode kann eine totale Ordnung hergestellt werden. Nach dem Einfügen eines Elements kann die Ordnung zerstört werden. Und natürlich muss der Comparer bemüht werden, woher sonst soll die Liste wissen, ob zwei Elemente 'gleich' sind? Das 'gleich' wird ja gerade durch den Comparer definiert. Was passiert, wenn Du keine Ordnungsrelation angibst? Geht das? |
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Zitat:
Delphi-Quellcode:
den Index des Elements in der Liste bekomme, also
IndexOf
Delphi-Quellcode:
Den Comparer habe ich jetzt erweitert zu
MyIndex := 4;
Assert( MyList.IndexOf( MyList[MyIndex] ) = MyIndex );
Delphi-Quellcode:
bzw. im originalen Code merke ich mir einfach den Index und der Kas ist gegessen.
LList := TObjectList<TItem>.Create( TComparer<TItem>.Construct(
function( const L, R : TItem ) : Integer begin Result := L.Value - R.Value; if Result = 0 then Result := Integer( L ) - Integer( R ); end ) ); |
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Wenn du über den Comparer gar nicht die Identität einer Instanz bestimmen willst, warum gibst du ihn der Liste mit?
Sortiere doch einfach nur danach und gut ist:
Delphi-Quellcode:
list.Sort(TComparer<TSample>.Construct(function(const l, r : TSample) : Integer
begin result := AnsiCompareStr(l.Text, r.Text); end)); |
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Zitat:
|
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Zitat:
|
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Zitat:
|
AW: Seltsames Verhalten von IndexOf bei sortierter generischer Liste
Zitat:
Delphi-Quellcode:
Generell würde ich aber empfehlen, in den Vergleich die ID mit aufzunehmen. Dann kannst du den Comparer beim Create schon übergeben.
LList.Sort( TComparer<TItem>.Construct(
function( const L, R : TItem ) : Integer begin Result := L.Value - R.Value; end ) ); |
Alle Zeitangaben in WEZ +1. Es ist jetzt 10:02 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