AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Generisch Maximum bestimmen

Ein Thema von freimatz · begonnen am 4. Apr 2023 · letzter Beitrag vom 7. Apr 2023
Antwort Antwort
Seite 1 von 2  1 2      
freimatz

Registriert seit: 20. Mai 2010
1.444 Beiträge
 
Delphi 11 Alexandria
 
#1

Generisch Maximum bestimmen

  Alt 4. Apr 2023, 17:00
Hallo,
wir haben immer wieder die Aufgabe ein Minimum oder Maximum zu bestimmen von einer Menge von Elementen in einem Container (TList<>, IList<>, TArray<>, etc.). Das ist keine grosse Sache - Schleife darüber, vorher Grenzwert festlegen und vergleichen.

Allerdings kam hier die Frage auf, ob man das nicht generisch lösen könnte.
Gefunden habe ich
etwas, das hilft aber nicht weiter.
Jemand meinte mit C# würde das gehen. Dort gäbe es eine Max function. Die Klasse müsste IEnumerable implementieren.

Die IList<T> von Spring4D hat eine Max function. Hat mir jemand ein Beispiel dazu?
Oder sonst irgendwelche Ideen oder Meinungen?
  Mit Zitat antworten Zitat
Benutzerbild von dummzeuch
dummzeuch

Registriert seit: 11. Aug 2012
Ort: Essen
1.604 Beiträge
 
Delphi 10.2 Tokyo Professional
 
#2

AW: Generisch Maximum bestimmen

  Alt 4. Apr 2023, 17:21
Jemand meinte mit C# würde das gehen. Dort gäbe es eine Max function. Die Klasse müsste IEnumerable implementieren.
Muss die Klasse dazu nicht auch irgendwie einen Wert liefern?

Aber sonst: Klar sollte das per Generics gehen.

for-Schleife via IEnumerable, für jedes Objekt den Wert ermitteln und dann vergleichen.

Das geht sogar ohne Compiler-Unterstützung für Generics, es reicht eine TInterfaceList und darin gespeicherte Interfaces, die den Wert liefern.
Thomas Mueller
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.453 Beiträge
 
Delphi 12 Athens
 
#3

AW: Generisch Maximum bestimmen

  Alt 4. Apr 2023, 18:05
Meinst du in etwa so? (Müsste man aber noch ausbauen)
Delphi-Quellcode:
type
  TGeneric = record
  public
    class function Max<T>(List: TEnumerable<T>): T; static;
  end;

class function TGeneric.Max<T>(List: TEnumerable<T>): T;
begin
  var comparer := IComparer<T>(TComparer<T>._Default);
  Result := Default(T);
  for var item in List do
    if comparer.Compare(Result, item) < 0 then
      Result := item;
end;
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
mytbo

Registriert seit: 8. Jan 2007
472 Beiträge
 
#4

AW: Generisch Maximum bestimmen

  Alt 4. Apr 2023, 19:30
Die IList<T> von Spring4D hat eine Max function. Hat mir jemand ein Beispiel dazu?
Delphi-Quellcode:
uses
  Spring.Collections;

var
  list: IList<Integer>;
begin
  list := TCollections.CreateList<Integer>;
  list.Add(1);
  list.Add(5);
  list.Add(3);
  ShowMessage(Format('%d, %d, %d, Max: %d',[list[0], list[1], list[2], list.Max]));
oder so:
Delphi-Quellcode:
var
  list: IList<Integer>;
begin
  list := TCollections.CreateList<Integer>([1, 5, 3]);
  ShowMessage(Format('%d, %d, %d, Max: %d',[list[0], list[1], list[2], list.Max]));
Bis bald...
Thomas

Geändert von mytbo ( 4. Apr 2023 um 19:38 Uhr) Grund: Beispiel erweitert
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

Registriert seit: 12. Aug 2003
Ort: Soest
4.016 Beiträge
 
Delphi 10.1 Berlin Enterprise
 
#5

AW: Generisch Maximum bestimmen

  Alt 4. Apr 2023, 23:41
Meinst du in etwa so? (Müsste man aber noch ausbauen)
Delphi-Quellcode:
type
  TGeneric = record
  public
    class function Max<T>(List: TEnumerable<T>): T; static;
  end;

class function TGeneric.Max<T>(List: TEnumerable<T>): T;
begin
  var comparer := IComparer<T>(TComparer<T>._Default);
  Result := Default(T);
  for var item in List do
    if comparer.Compare(Result, item) < 0 then
      Result := item;
end;
Da ist nen kleiner aber subtiler Bug drin: wenn dein TEnumerable<T> die Zahlen -1, -3 und -7 enthält, dann würde es inkorrekterweise 0 als Max ausgeben und nicht -1.
Result muss mit dem ersten Wert aus der Sequenz initialisiert werden und ab da muss verglichen werden. Außerdem brauchts ein Verhalten, wenn die Sequenz leer ist (in Spring gibt's genau wie in .NET ne Exception)
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.453 Beiträge
 
Delphi 12 Athens
 
#6

AW: Generisch Maximum bestimmen

  Alt 5. Apr 2023, 01:23
Da ist nen kleiner aber subtiler Bug drin:
Ich weiß - deswegen mein Hinweis mit dem ausbauen. Es ging mir im Wesentlichen erstmal um die Syntax.

Eine andere Tretmine: was wenn für <T> kein Default-Comparer existiert?
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
freimatz

Registriert seit: 20. Mai 2010
1.444 Beiträge
 
Delphi 11 Alexandria
 
#7

AW: Generisch Maximum bestimmen

  Alt 5. Apr 2023, 08:08
Sorry, das ist keine Tretmine, sondern der Normalfall bei uns.
Tut mir leid hätte ich vielleicht gleich am Anfang noch dazu schreiben sollen. Bei "Menge von Elementen" handelt es sich nicht nur um Integer sondern Records, Klassen, etc. z.B. einem record von einem Abstand: Double und einem vector.
Müsste dann der Record einen IComparer implementerien? Oder die Liste. Habt ihr mir da auch ein Beispiel dazu? Basierend auf Spring4D reicht mir mal vorläufig.
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.453 Beiträge
 
Delphi 12 Athens
 
#8

AW: Generisch Maximum bestimmen

  Alt 5. Apr 2023, 09:29
Wie bei anderen Lösungen auch, muss der Comparer dann in irgendeiner Form bereitgestellt werden. Das kann durch einen zusätzlichen Parameter geschehen oder durch eine Art Registrierung zu den verschiedenen Typen.
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

Registriert seit: 12. Aug 2003
Ort: Soest
4.016 Beiträge
 
Delphi 10.1 Berlin Enterprise
 
#9

AW: Generisch Maximum bestimmen

  Alt 5. Apr 2023, 10:33
was wenn für <T> kein Default-Comparer existiert?
Es existiert immer einer (bzw wird erstellt) - ob ein binary compare eines records Sinn ergibt, ist dann eine andere Geschichte.

In Spring 2.0 bin ich einen Schritt (oder mehrere) weiter gegangen als die RTL, indem ich die Möglichkeit gegeben habe, einen Default Comparer zu registrieren (siehe unit Spring.Comparers).

Übrigens bieten alle Methoden der Spring Collections, die irgendwas mit comparern machen, immer einen Overload an, in den man einen comparer oder eine comparison übergeben kann, sollte man das nicht bei der Erstellung der Liste z.B. schon getan haben.

Delphi-Quellcode:
function CompareMyRecord(const left, right: TMyRecord): Integer;
begin
  // Vergleichslogik
end;

var
  list: IList<TMyRecord>;
  maxRec: TMyRecord;
begin
  ...
  maxRec := list.Max(CompareMyRecord);
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight

Geändert von Stevie ( 5. Apr 2023 um 10:50 Uhr)
  Mit Zitat antworten Zitat
mytbo

Registriert seit: 8. Jan 2007
472 Beiträge
 
#10

AW: Generisch Maximum bestimmen

  Alt 5. Apr 2023, 18:44
Müsste dann der Record einen IComparer implementerien? Oder die Liste. Habt ihr mir da auch ein Beispiel dazu? Basierend auf Spring4D reicht mir mal vorläufig.
Zur Veranschaulichung der Fälle. Kombination ist möglich. Beispiel für TComparison<T>:
Delphi-Quellcode:
type
  TRec = record
    x, y: Integer;
  end;

function CompareRecY(const pmcLeft, pmcRight: TRec): Integer;
begin
  Result := (pmcLeft.y - pmcRight.y);
end;

const
  RECS: array[0..2] of TRec = ((x: 1; y: 4), (x: 5; y: 6), (x: 3; y: 2));
var
  list: IList<TRec>;
begin
  list := TCollections.CreateList<TRec>(RECS);
  list.Sort(CompareRecY);
  ShowMessage(Format('First Y: %d, Max Y: %d',[list[0].y, list.Max(CompareRecY).y]));
end;
Beispiel für IComparer<T>:
Delphi-Quellcode:
type
  TData = class
  private
    FX, FY: Integer;
  public
    constructor Create(pmX, pmY: Integer);
    property X: Integer
      read FX;
    property Y: Integer
      read FY;
  end;

  TDataYComparer<T: TData> = class(TComparer<T>)
  public
    function Compare(const pmcLeft, pmcRight: T): Integer; override;
  end;

constructor TData.Create(pmX, pmY: Integer);
begin
  inherited Create;
  FX := pmX;
  FY := pmY;
end;

function TDataYComparer<T>.Compare(const pmcLeft, pmcRight: T): Integer;
begin
  Result := (T(pmcLeft).Y - T(pmcRight).Y);
end;

var
  list: IList<TData>;
  comp: IComparer<TData>;
begin
  comp := TDataYComparer<TData>.Create;
  list := TCollections.CreateObjectList<TData>(comp, True);
  list.Add(TData.Create(1, 4));
  list.Add(TData.Create(5, 6));
  list.Add(TData.Create(3, 2));
  list.Sort;
  ShowMessage(Format('First Y: %d, Max Y: %d',[list[0].Y, list.Max.Y]));
end;
Bis bald...
Thomas
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 03:46 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