Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen? (https://www.delphipraxis.net/191032-enum-string-mehrere-overloaded-funktionen-zu-einer-einzigen-zusammenfassen.html)

a.def 1. Dez 2016 09:56

Delphi-Version: 5

Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Folgende Funktion befindet sich in meinem Code und davon eine für jedes meiner Enums
Delphi-Quellcode:
// EnumGetString(TWProcesses.wpIdle) würde string 'wpIdle' zurückgeben
function EnumGetString(aEnumValue: TWProcesses): string;
var
 bVal: Byte;
begin
 Move(aEnumValue, bVal, SizeOf(TWProcesses));
 Result := GetEnumName(TypeInfo(TWProcesses), bVal);
end;
Ist es möglich diese Funktion so abzuändern, dass ich alle meine Enums an eine einzige Funktion schicken kann, welche mir dann den String zurückgibt wie oben zu sehen auch?

Der schöne Günther 1. Dez 2016 09:58

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Generics

http://docwiki.embarcadero.com/RADSt...BCber_Generics

Uwe Raabe 1. Dez 2016 10:36

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von Der schöne Günther (Beitrag 1355093)
Generics

Leider gibt es kein Constraint für Enums bei einem generischen Typ. Man verliert dabei also die Typsicherheit.

Der schöne Günther 1. Dez 2016 10:44

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Man kann den Compiler nicht zufällig anweisen keine Schlussfolgerungen über den generischen Typ anzustellen sodass der Benutzer den generischen Typ angeben muss?

Alternativ kann man, wenn einen das stört, ja einen Integer übergeben und steckt in seine "EnumZuString"-Methode dann nicht mehr "myEnum" sondern "Ord(myEnum)" rein.

Uwe Raabe 1. Dez 2016 11:08

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von Der schöne Günther (Beitrag 1355102)
Man kann den Compiler nicht zufällig anweisen keine Schlussfolgerungen über den generischen Typ anzustellen sodass der Benutzer den generischen Typ angeben muss?

Das habe ich jetzt nicht verstanden.


Zitat:

Zitat von Der schöne Günther (Beitrag 1355102)
Alternativ kann man, wenn einen das stört, ja einen Integer übergeben und steckt in seine "EnumZuString"-Methode dann nicht mehr "myEnum" sondern "Ord(myEnum)" rein.

Dann fehlt aber die TypeInfo, die man für GetEnumName braucht.

Der schöne Günther 1. Dez 2016 11:22

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von Uwe Raabe (Beitrag 1355107)
Das habe ich jetzt nicht verstanden.

Ich meine dass man sich eine generische Funktion schreibt bei welcher man den generischen Typen angeben MUSS. Dann hat man den Typen ja und kann sich seine TypInformation holen.

Für den Fall dass ein Enum nicht mehr als 128 (oder 256?) Einträge enthält ginge ja so etwas hier, aber perfekt ist das auch nicht :|. Mein Favorit ist der zweite der drei Aufrufe.


Delphi-Quellcode:
program Project16;

{$APPTYPE CONSOLE}

{$R *.res}

uses System.TypInfo;

type
   TEnum = record
      public class function GetName<T>(const enum: T): String; overload; static;
      public class function GetName<T>(const ordinalValue: System.ShortInt): String; overload; static;
   end;

{ TEnum }

class function TEnum.GetName<T>(const enum: T): String;
var
   ordinalValue: Integer;
begin
   ordinalValue := Default(Integer);
   Move(enum, ordinalValue, SizeOf(T));

   Result := GetEnumName( TypeInfo(T), ordinalValue );
end;

class function TEnum.GetName<T>(const ordinalValue: System.ShortInt): String;
begin
   Result := GetEnumName( TypeInfo(T), ordinalValue );
end;

type
   TMyEnum = (uno, dos, tres);
var
   myEnum:   TMyEnum;
begin
   myEnum := TMyEnum.dos;

   // Nicht typsicher, myEnum könnte genauso gut ein Float sein
   WriteLn( TEnum.GetName<TMyEnum>(myEnum) );

   // Typsicher, WENN man sich zwingt EXPLIZIT "TMyEnum" anzugeben
   WriteLn( TEnum.GetName<TMyEnum>(myEnum) );

   // Sieht typischer aus, aber statt myEnum könnte man ebenso gut "42" reinstecken
   WriteLn( TEnum.GetName<TMyEnum>( Ord(myEnum) ) );

   ReadLn;
end.

Zacherl 1. Dez 2016 11:36

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von Der schöne Günther (Beitrag 1355102)
Man kann den Compiler nicht zufällig anweisen keine Schlussfolgerungen über den generischen Typ anzustellen sodass der Benutzer den generischen Typ angeben muss?

Einen Typ angeben musst du bei einer generischen Klasse ja sowieso zwingend. Das Problem ist, dass die Delphi Generics außer
Delphi-Quellcode:
class
und
Delphi-Quellcode:
interface
keine Constraints zulassen. Wobei Constraints eh nur ein Versuch sind ein viel prägnanteres Design-Flow abzuschwächen:

Delphi-Quellcode:
class procedure TGenericClass<TEnum>.Print(Enum: TEnum);
var
  E: TEnum;
begin
  for E := Low(Enum) to High(Enum) do
  begin
    WriteLn(Ord(Enum));
  end;
end;
An dieser Stelle meldet sich der Delphi-Compiler obwohl ich die Klasse nirgends verwende mit einem Fehler, weil er einfach so davon ausgeht, dass
Delphi-Quellcode:
TEnum
kein Enum-Typ ist, sondern eine Klasse (bzw. untypisiert ist, oder was der Compiler auch immer an dieser Stelle als Standardtyp annimt). Dies ist mir relativ unverständlich, wenn ich mal mit den Templates in C++ vergleiche (diese sind auf der trivialsten Ebene praktisch Generics):

Der Code in Template-Funktionen wird erst dann verifiziert, sobald er auch generiert wird. Und generiert wird er einmalig für jeden distinkten Datentyp den ich tatsächlich irgendwo im Code an die Template-Klasse übergebe.
Delphi-Quellcode:
TGenericClass<TIrgendeinExistierendesEnum>.Print(MyEnum)
würde also ohne Probleme funktionieren, während
Delphi-Quellcode:
TGenericClass<TIrgendeineKlasse>.Print(MyClassInstance)
korrekterweise den Fehler erzeugt, dass man
Delphi-Quellcode:
Low
natürlich nicht auf einen Klassentyp anwenden darf.

Die C++ Templates sind im Gegensatz zu den Delphi Generics sozusagen Context-aware.

Am Ende kommt jedenfalls bei raus, dass man unter Delphi gezwungen wird ziemlich viele nicht typsichere Operationen durchzuführen, bei denen man höchstens mit viel Mühe und unter Verwendung der RTTI eine teilweise Typsicherheit wiederherstellen kann:
http://www.delphipraxis.net/190944-r...alisieren.html

Edit: @Günther: Und wer hindert dich bei deiner Klasse daran einfach
Delphi-Quellcode:
TEnum.GetName<TForm1>(Form1)
aufzurufen?

a.def 1. Dez 2016 11:40

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Die Antworten sind ja schon erstaunlich.
Wäre es bei so vielen Komplikationen einen schönen Code zu bekommen nicht die einfachste Möglichkeit mehrere overloaded Funktionen zu fahren?

Zacherl 1. Dez 2016 11:58

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von a.def (Beitrag 1355119)
Wäre es bei so vielen Komplikationen einen schönen Code zu bekommen nicht die einfachste Möglichkeit mehrere overloaded Funktionen zu fahren?

Wir wollten dich nicht verunsichern, sondern nur auf eine mögliche Gefahr hinweisen. Ich würde es trotzdem folgendermaßen lösen:
Delphi-Quellcode:
type
  TEnumHelper<TEnum> = record
    public class function GetName(Value: TEnum): String; overload; static;
  end;

{ TEnumHelper<TEnum> }

class function TEnumHelper<TEnum>.GetName(Value: TEnum): String;
var
  TypInfo: PTypeInfo;
  TypData: PTypeData;
  V: Integer;
begin
  TypInfo := TypeInfo(TEnum);
  {$IFDEF DEBUG}
  if (TypInfo^.Kind <> tkEnumeration) then
  begin
    raise Exception.Create('Invalid generic type.');
  end;
  {$ENDIF}
  TypData := GetTypeData(TypInfo);
  case TypData^.OrdType of
    otSByte,
    otUByte:
      V := PByte(@Value)^;
    otSWord,
    otUWord:
      V := PWord(@Value)^;
    otSLong,
    otULong:
      V := PInteger(@Value)^;
  end;
  Result := GetEnumName(TypeInfo(TEnum), V);
end;
Du solltest halt nur aufpassen, dass du dieser Klasse als Typ tatsächlich nur Enums übergibst und nichts anderes. Im Debug-Mode würdest du zwar im Zweifelsfalle eine Runtime-Exception bekommen, aber bei Code-Pfaden, die nicht oft ausgeführt werden, entdeckt man so einen Laufzeitfehler teilweise erst recht spät.

Alternativ müsste auch das hier gehen:
Delphi-Quellcode:
class function TEnumHelper<TEnum>.GetName(Value: TEnum): String;
var
  TypInfo: PTypeInfo;
  V: Integer;
begin
  TypInfo := TypeInfo(TEnum);
  {$IFDEF DEBUG}
  if (TypInfo^.Kind <> tkEnumeration) then
  begin
    raise Exception.Create('Invalid generic type.');
  end;
  {$ENDIF}
  case SizeOf(TEnum) of
    1: V := PByte(@Value)^;
    2: V := PWord(@Value)^;
    4: V := PInteger(@Value)^;
  end;
  Result := GetEnumName(TypeInfo(TEnum), V);
end;

uligerhardt 1. Dez 2016 14:21

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Dumme Frage: Warum nimmst du statt dem:
Zitat:

Zitat von Zacherl (Beitrag 1355123)
Delphi-Quellcode:
  {$IFDEF DEBUG}
  if (TypInfo^.Kind <> tkEnumeration) then
  begin
    raise Exception.Create('Invalid generic type.');
  end;
  {$ENDIF}

nicht einfach Assert?
Delphi-Quellcode:
  Assert(TypInfo^.Kind = tkEnumeration, 'Invalid generic type.');

Zacherl 1. Dez 2016 16:36

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von uligerhardt (Beitrag 1355150)
Dumme Frage: Warum nimmst du nicht einfach Assert?
Delphi-Quellcode:
  Assert(TypInfo^.Kind = tkEnumeration, 'Invalid generic type.');

Guter Einwand :thumb: Keine Ahnung, warum ich das so umständlich gemacht hatte.

a.def 1. Dez 2016 16:56

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Kann man diese Funktion auch in mein aktuell bestehendes Konzept einbauen?

Delphi-Quellcode:
unit enum_functions;

uses ;

type
 TEnumFunctions = class
 private
  //
 public
  // meine Funktionen
 end;

implementation

// meine Funktionen

end;
Delphi-Quellcode:
var _enumFunctions: TEnumFunctions;

// Aufruf meiner Funktionen mit
_enumFunctions.Funktionsname();

Zacherl 1. Dez 2016 18:21

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von a.def (Beitrag 1355186)
Kann man diese Funktion auch in mein aktuell bestehendes Konzept einbauen?

Natürlich kannst du dir die Funktion auch in deine bestehende Klasse einbauen. Wenn du dir allerdings Sorgen machst, dass du bei der Helper Klasse von mir zusätzliche Instanzen pflegen musst, dann kann ich dich beruhigen, da du die Funktion ganz einfach in der Form
Delphi-Quellcode:
S := TEnumHelper<TMyEnum>.GetName(EnumValue)
aufrufen kannst.

In deiner bestehender Klasse müsstest du dann die Funktion selbst mit Generics ausrüsten, anstelle der ganzen Klasse (sofern die anderen Helper nicht auch typisiert werden sollen).

Generell würde ich dir allerdings raten deine
Delphi-Quellcode:
TEnumFunctions
mit
Delphi-Quellcode:
class function Bla: X; static
bzw.
Delphi-Quellcode:
class procedure Bla; static
auszurüsten statt der normalen Methoden. Dann benötigst du auch dafür keine Instanz mehr, was die Aufrufe zum einen weniger umständlich macht und dir später im Kompilat sogar ein klein wenig (nagut nur sehr sehr wenig - aber immerhin :stupid:) Performance-Overhead einspart.
Letzter Tipp natürlich unter dem Vorbehalt, dass du wirklich keine Instanz (also z.b. Zugriff auf Klassen-Felder, persistenter State, etc.) benötigst.

a.def 1. Dez 2016 19:09

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von Zacherl (Beitrag 1355188)
Generell würde ich dir allerdings raten deine
Delphi-Quellcode:
TEnumFunctions
mit
Delphi-Quellcode:
class function Bla: X; static
bzw.
Delphi-Quellcode:
class procedure Bla; static
auszurüsten statt der normalen Methoden. Dann benötigst du auch dafür keine Instanz mehr, was die Aufrufe zum einen weniger umständlich macht und dir später im Kompilat sogar ein klein wenig (nagut nur sehr sehr wenig - aber immerhin :stupid:) Performance-Overhead einspart.

Daran hatte ich schon lange gedacht. Denn mit meiner aktuellen Klasse+Instanz kann ich nur Funktionen und Prozeduren einbauen, aber keine private und public Variablen.
Nur ich weiß leider nicht wie das geht :(

Zitat:

Letzter Tipp natürlich unter dem Vorbehalt, dass du wirklich keine Instanz (also z.b. Zugriff auf Klassen-Felder, persistenter State, etc.) benötigst.
Das habe ich nicht ganz verstanden. Aber ich glaube du meinst, dass man dann keine Variablen im TEnumFunctions = class-Konstrukt benutzen kann.

Edit: habs glaube ich

Delphi-Quellcode:
type
 TEnumFunctions = class
 private
  //
 public
  class function TestFunction(sInput: string): string; static;
 end;

implementation

class function TEnumFunctions.TestFunction(sInput: string): string;
begin
 Result := sInput + ' - ZUSATZ';
end;

// Aufruf
ShowMessage(TEnumFunctions.TestFunction('123'));
... bin gerade schon alles fleißig am umstellen, damit ich die Instanzen-"Variable" loswerde.

Sollte dann lieber folgendes bleiben...
Delphi-Quellcode:
type
 TEnumFunctions = class
Oder besser
Delphi-Quellcode:
type
 TEnumFunctions = record
?

Zacherl 1. Dez 2016 23:15

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Zitat:

Zitat von a.def (Beitrag 1355189)
Sollte dann lieber folgendes bleiben...
Delphi-Quellcode:
type
 TEnumFunctions = class
Oder besser
Delphi-Quellcode:
type
 TEnumFunctions = record

Tatsächlich verwenden die meisten Leute - mich eingeschlossen - lieber einen Record, wenn es eh nur statische Methoden gibt. Kann dir aber grade gar nicht genau sagen, was dazu die Überlegung im Hintergrund ist. Ich rede mir ein, dass der Compiler so weniger RTTI-"Müll" generiert.

bepe 2. Dez 2016 05:57

AW: Enum in String mehrere Overloaded Funktionen zu einer einzigen zusammenfassen?
 
Hi,

es gibt noch eine andere Möglichkeit die auch mit nicht Enums kein Problem hätte: TValue. Das nutze ich immer.

Delphi-Quellcode:
...
uses
  System.RTTI
...
class function TEnumHelper<TEnum>.GetName(Value: TEnum): String;
begin
  Result := TValue.From<TEnum>(Value).ToString;
end;


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