Delphi-PRAXiS
Seite 2 von 3     12 3      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Algorithmen, Datenstrukturen und Klassendesign (https://www.delphipraxis.net/78-algorithmen-datenstrukturen-und-klassendesign/)
-   -   Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren (https://www.delphipraxis.net/216198-case-anweisung-zwingen-alle-elemente-des-typen-auszuprogrammieren.html)

lxo 19. Nov 2024 12:52

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Meine Idee mit dem Ord(High(TEnum)) war so gedacht:

Delphi-Quellcode:
type
  TEnum = ( t1, t2, t3);
var
  lEnum: TEnum;
begin
  {$DEFINE ENUMCOUNT 0}
  case lEnum of
    t1:
     begin
      {$DEFINE ENUMCOUNT ENUMCOUNT+1}
      ...
     end;
    t2:
     begin
      {$DEFINE ENUMCOUNT ENUMCOUNT+1}
      ...
     end;
    // Hier immer letztes Element aus der Case-Anweisung eintragen.
    {$IF Ord(High(TEnum)) <> ENUMCOUNT} {$MESSAGE Error 'Case-Fall fuer jedes Element ausprogrammieren!'} {$IFEND}
  else
    raise Exception.Create('Fehlermeldung');
  end;
end;

sh17 19. Nov 2024 13:05

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von himitsu (Beitrag 1543239)
Das ist so aber auch schwer mitzubekommen,
bzw. es wird nicht lesbarer. :stupid:
Delphi-Quellcode:
{$IF Ord(High(TEnum)) <> 1} {$MESSAGE Error 'peng'} {$IFEND}


Delphi-Quellcode:
array[TEnum] of ... = (...);

Hier bekommst du die Änderung der Anzahl mit.

Wenn jemand aber gleichzeitig etwas hinzufügt und löscht,
ohne dass es zwischendrin angepasst wurde,
dann flutscht das natürlich durch.

Dann würde es aber auch knallen, weil ein alter Wert fehlt - wenn man sich bisher an seine Regel der vollständigen Auswertung gehalten hat.

dummzeuch 19. Nov 2024 16:42

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von lxo (Beitrag 1543238)
Zitat:

Zitat von dummzeuch (Beitrag 1543237)
Zitat:

Zitat von lxo (Beitrag 1543236)
Zitat:

Zitat von himitsu (Beitrag 1543234)
Delphi-Quellcode:
    {$IF High(TEnum) <> t2} {$MESSAGE Error 'peng'} {$IFEND}


Vielleicht ganz banal:

Delphi-Quellcode:
{$IF Ord(High(TENum) <> 1}{$MESSAGE Error 'peng'} {$IFEND}

Ist doch erstmal das gleiche wie von himitsu oder wo soll da der Unterschied sein?

Es erkennt auch, wenn jemand einen enum-Wert einfügt, nicht nur wenn jemand einen Wert anhängt. Bei beidem ändert sich Ord(High(TENum)), aber nur bei letzterem ändert sich High(TENum).

Der schöne Günther 19. Nov 2024 17:10

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Siehe auch:
Delphi merkt nicht wenn ein case-Statement vollständig ist, oder?

und

Zitat:

Zitat von Der schöne Günther (Beitrag 1498589)
Bei Sprachen wie Swift oder Rust bricht der Compiler mit einem Fehler ab, wenn ein match-Statement nicht vollständig ist. Delphi bekommt von all dem aber nichts mit(...)


Rollo62 19. Nov 2024 17:19

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von lxo (Beitrag 1543235)
Hätte es halt gerne direkt beim kompilieren.

Zitat:

Zitat von lxo (Beitrag 1543226)
Hallo,
ich möchte bei einer Case-Anweisung einen Compiler-Fehler bekommen ....

Genau, beim Compilieren kommt der Fehler mit so einer Lösung
Delphi-Quellcode:
CEnum : array[ TEnum ] of String = ( 't1', 't2', 't3' ); //Knallt schon hier, wenn es nicht übereinstimmt
Dass man das jetzt erweitern muss, ist ja genau Sinn des Crashes.
Nebenbei hätte so eine noch weitere Vorteile, also zum Beispiel Textausgabe, Rückgabe beliebiger Werte, etc.

Eventuell sucht Du ja sowas, etwas "Verboseres":

Delphi-Quellcode:
type
  TMyEnum = (Value0, Value1, Value2);

const
  ExpectedEnumSize = 3; // Erwartete Anzahl der Elemente in TMyEnum

{$IF    Ord(High(TMyEnum)) > (ExpectedEnumSize-1) }
  {$MESSAGE ERROR 'TMyEnum wurde erweitert!'      }
{$ELSEIF Ord(High(TMyEnum)) < (ExpectedEnumSize-1) }
  {$MESSAGE ERROR 'TMyEnum wurde verkleinert!'    }
{$IFEND}

dummzeuch 19. Nov 2024 17:29

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von Rollo62 (Beitrag 1543254)
Genau, beim Compilieren kommt der Fehler mit so einer Lösung
Delphi-Quellcode:
CEnum : array[ TEnum ] of String = ( 't1', 't2', 't3' ); //Knallt schon hier, wenn es nicht übereinstimmt

Das Problem ist, dass man dann aber die betroffenen case-Statements alle manuell raussuchen muss, was nicht ganz so einfach ist, da dort ja nicht der Typ sondern eine Variable des Typs verwendet wird, man kann also nicht einfach ein grep auf den Namen des Typs machen. Wenn der Compiler bei jedem davon einen Fehler schmeißt, ist das einfacher. Kann ja auch gerne abschaltbar sein (wobei man das ja auch mit einem else erreichen könnte).

himitsu 19. Nov 2024 18:29

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Er muß ja keinen Fehler werfen, aber z.B. eine Warnung oder Info wäre eigentlich problemlos möglich, wenn bei einem Enum nicht alle Werte benutzt wurden.

Ja, natürlich auch abschaltbar, bzw. vielleicht über ein Attribut oder Keyword, explizit anforderbar,
denn es kommt vor, dass ich in manchen Case absichtlich garnicht alles verwende,
z.B. wenn an einer Stelle nur ein Teil benötigt wird und der Rest einfach nichts machen soll.

Uwe Raabe 19. Nov 2024 20:43

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von dummzeuch (Beitrag 1543255)
da dort ja nicht der Typ sondern eine Variable des Typs verwendet wird, man kann also nicht einfach ein grep auf den Namen des Typs machen.

Wenn man SCOPEDENUMS verwendet, schon...

bernau 20. Nov 2024 07:43

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von Uwe Raabe (Beitrag 1543272)
Wenn man SCOPEDENUMS verwendet, schon...

Wenn da nicht mein 25 Jahre alter Code wäre. :oops:

dummzeuch 20. Nov 2024 08:05

AW: Case-Anweisung - Zwingen alle Elemente des Typen auszuprogrammieren
 
Zitat:

Zitat von himitsu (Beitrag 1543256)
Ja, natürlich auch abschaltbar, bzw. vielleicht über ein Attribut oder Keyword, explizit anforderbar, denn es kommt vor, dass ich in manchen Case absichtlich garnicht alles verwende,
z.B. wenn an einer Stelle nur ein Teil benötigt wird und der Rest einfach nichts machen soll.

Dafür gibt's ja schon else. Dann muss man allerdings evtl. anderen Tools beibringen, dass ein leeres else durchaus OK ist.

Delphi-Quellcode:
case bla of
  b1: HandleB1;
  b2: HandleB2;
else
  // nix tun
end;


Alle Zeitangaben in WEZ +1. Es ist jetzt 17:34 Uhr.
Seite 2 von 3     12 3      

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