Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Sonstige Fragen zu Delphi (https://www.delphipraxis.net/19-sonstige-fragen-zu-delphi/)
-   -   Delphi C- union Struktur in Delphi (https://www.delphipraxis.net/45158-c-union-struktur-delphi.html)

jackie 30. Apr 2005 18:15


C- union Struktur in Delphi
 
Hallo!
Kann ich in Delphi union Strukturen erstellen (wie in C)?
Beispiel:
Code:
...
union {
    ULONGLONG Alignment;
    struct {
      ULONG Length;
      DWORD IfIndex;
    };
  };
...

rantanplan99 30. Apr 2005 18:45

Re: C- union Struktur in Delphi
 
Googlest du, findest du diesen link hier.

Zitat:

Variant Records: The equivalent to the C-union structure

Is there a way to create a C 'union'-like structure in Delphi? That is, a structure that uses the same memory area?

The Delphi (Pascal/ObjectPascal) equivalent to a C-union structure is called a Variant Record (not to be confused with the Variant "type" available in Delphi 2.0+). As with a C-union, the Pascal variant record allows several structure types to be combined into one, and all will occupy the same memory space. Look up the syntax declaration under "Records" in the help file. But here's an example:
Delphi-Quellcode:
type
 TPerson = record
   FirstName, LastName: string[40];
   BirthDate: TDate;
   case Citizen: Boolean of
     True: (BirthPlace: string[40]);
     False: (Country: string[20];
       EntryPort: string[20];
       EntryDate: TDate;
       ExitDate: TDate);
 end;
The record above is actually a single expression of two records that could describe a person:
Delphi-Quellcode:
type
 TPersonCitizen = record
   FirstName, LastName: string[40];
   BirthDate: TDate;
   BirthPlace: string[40]
 end;
and
Delphi-Quellcode:
type
 TPersonAlien = record
   FirstName, LastName: string[40];
   BirthDate: TDate;
   Country: string[20];
   EntryPort: string[20];
   EntryDate: TDate;
   ExitDate: TDate;
 end;
And as in a union, the combination of the two types of records makes for much more efficient programming, because a person could be expressed in a variety of ways.

Everything I explained above is pretty hypothetical stuff. In Delphi, the TRect structure that describes a rectangle is actually a variant record:
Delphi-Quellcode:
type
TPoint = record
  X: Longint;
  Y: Longint;
end;

TRect = record
  case Integer of
    0: (Left, Top, Right, Bottom: Integer);
    1: (TopLeft, BottomRight: TPoint);
  end;
where the coordinates of the rectangle can be expressed using either four integer values or two TPoints.

I realize this is pretty quick and dirty, so I suggest you refer to the help file for a more in-depth explanation, or go to your nearest book store or library and look at any Pascal book (not Delphi -- most won't explain this fairly esoteric structure). However, if you're familiar with the C-union, this stuff should be an absolute breeze.

Copyright © 1997 Brendan V. Delumpa All Rights Reserved
Copyright © The Delphi Corner 2001 All Rights Reserved
rantanplan

Christian Seehase 30. Apr 2005 18:46

Re: C- union Struktur in Delphi
 
Moin Gregor,

das sollte sich grundsätzlich über case lösen lassen.

In Deinem Beispiel müsste das so aussehen:

Delphi-Quellcode:
type
  TMyUnion = packed record
    case byte of
      1 : (Alignment : int64;)
      2 : (Length   : DWORD;
           IfIndex  : DWORD;)
  end;
Schau doch mal in der Hilfe unter dem Stichwort "variante Records".

Ein gravierendes Problem gibt es allerdings auf jeden Fall:
Der variante Teil muss am Ende der Record-Deklaration stehen, so dass man ziemliche Verrenkungen machen muss, wenn man so eine Struktur mittendrin braucht.

jackie 1. Mai 2005 15:06

Re: C- union Struktur in Delphi
 
Hallo!
Also erstmal danke für die Antworten.
Die originale Struktur (C) sieht so aus:
Code:
typedef struct _IP_ADAPTER_ADDRESSES {
  union {
    ULONGLONG Alignment;
    struct {
      ULONG Length;
      DWORD IfIndex;
    };
  };
  struct _IP_ADAPTER_ADDRESSES* Next;
  PCHAR AdapterName;
  PIP_ADAPTER_UNICAST_ADDRESS FirstUnicastAddress;
  PIP_ADAPTER_ANYCAST_ADDRESS FirstAnycastAddress;
  PIP_ADAPTER_MULTICAST_ADDRESS FirstMulticastAddress;
  PIP_ADAPTER_DNS_SERVER_ADDRESS FirstDnsServerAddress;
  PWCHAR DnsSuffix;
  PWCHAR Description;
  PWCHAR FriendlyName;
  BYTE PhysicalAddress[MAX_ADAPTER_ADDRESS_LENGTH];
  DWORD PhysicalAddressLength;
  DWORD Flags;
  DWORD Mtu;
  DWORD IfType;
  IF_OPER_STATUS OperStatus;
  DWORD Ipv6IfIndex;
  DWORD ZoneIndices[16];
  PIP_ADAPTER_PREFIX FirstPrefix;
} IP_ADAPTER_ADDRESSES,
*PIP_ADAPTER_ADDRESSES;
Ich habs jetzt so übersetzt:
Delphi-Quellcode:
PIP_ADAPTER_ADDRESSES = ^TIP_ADAPTER_ADDRESSES;
  TIP_ADAPTER_ADDRESSES = packed record
      {struct _IP_ADAPTER_ADDRESSES* Next;}
      AdapterName: PChar;
      FirstUnicastAddress: PIP_ADAPTER_UNICAST_ADDRESS;
      FirstAnycastAddress: PIP_ADAPTER_ANYCAST_ADDRESS;
      FirstMulticastAddress: PIP_ADAPTER_MULTICAST_ADDRESS;
      FirstDnsServerAddress: PIP_ADAPTER_DNS_SERVER_ADDRESS;
      DnsSuffix: PWCHAR;
      Description: PWCHAR;
      FriendlyName: PWCHAR;
      PhysicalAddress: Byte[MAX_ADAPTER_ADDRESS_LENGTH];
      PhysicalAddressLength: DWord;
      Flags: DWord;
      Mtu: DWord;
      IfType: DWord;
      OperStatus: IF_OPER_STATUS;
      Ipv6Index: DWord;
      ZoneIndices: DWORD[16];
      FirstPrefix: PIP_ADAPTER_PREFIX;
      {union??}
    end;
Bei den Sachen in Klammern bin ich mir nicht sicher. Ich weis auch nicht, welchen Namen ich dem union geben soll?!
Zitat:

Der variante Teil muss am Ende der Record-Deklaration stehen, so dass man ziemliche Verrenkungen machen muss, wenn man so eine Struktur mittendrin braucht.
Das sollte hier aber eigentlich kein Problem sein, oder?

Mephistopheles 1. Mai 2005 15:20

Re: C- union Struktur in Delphi
 
Wie schreibst du so schön in deiner Signatur? :zwinker: Wurde doch oben alles erklärt.


Delphi-Quellcode:
  PIP_ADAPTER_ADDRESSES = ^TIP_ADAPTER_ADDRESSES;
  TIP_ADAPTER_ADDRESSES = packed record
    case Integer of
      1: (Alignment: ULONGLONG);
      2: (Length: ULONG; IfIndex: DWORD; // <-- matches sizeof(ULONGLONG)
        Next: PIP_ADAPTER_ADDRESSES;    // <-- start of variant part
        AdapterName: PChar;
        FirstUnicastAddress: PIP_ADAPTER_UNICAST_ADDRESS;
        FirstAnycastAddress: PIP_ADAPTER_ANYCAST_ADDRESS;
        FirstMulticastAddress: PIP_ADAPTER_MULTICAST_ADDRESS;
        FirstDnsServerAddress: PIP_ADAPTER_DNS_SERVER_ADDRESS;
        DnsSuffix: PWCHAR;
        Description: PWCHAR;
        FriendlyName: PWCHAR;
        PhysicalAddress: Byte[MAX_ADAPTER_ADDRESS_LENGTH];
        PhysicalAddressLength: DWORD;
        Flags: DWORD;
        Mtu: DWORD;
        IfType: DWORD;
        OperStatus: IF_OPER_STATUS;
        Ipv6Index: DWORD;
        ZoneIndices: DWORD[16];
        FirstPrefix: PIP_ADAPTER_PREFIX;
        )
  end;
So müßte es aussehen, was bei mir aber nicht kompiliert, wegen DWORD[16] und Byte[MAX_ADAPTER_ADDRESS_LENGTH]. Kann man umgehen, indem man dafür auch eigene Typen einführt.

jackie 1. Mai 2005 15:22

Re: C- union Struktur in Delphi
 
Danke!

Christian Seehase 1. Mai 2005 15:23

Re: C- union Struktur in Delphi
 
Moin Gregor,

das Problem ist, dass in der Originalstruktur die Union am Anfang steht, die kannst Du jetzt also nicht einfach an das Ende setzen.

Ich denke mal so müsste es gehen:

Code:
  PIP_ADAPTER_ADDRESSES = ^TIP_ADAPTER_ADDRESSES;
  TIP_ADAPTER_ADDRESSES = packed record
    case byte of
      0 : (Alignment : int64;
           Next : P_IP_ADAPTER_ADDRESSES;
           AdapterName: PChar;
           FirstUnicastAddress: PIP_ADAPTER_UNICAST_ADDRESS;
           FirstAnycastAddress: PIP_ADAPTER_ANYCAST_ADDRESS;
           FirstMulticastAddress: PIP_ADAPTER_MULTICAST_ADDRESS;
           FirstDnsServerAddress: PIP_ADAPTER_DNS_SERVER_ADDRESS;
           DnsSuffix: PWCHAR;
           Description: PWCHAR;
           FriendlyName: PWCHAR;
           PhysicalAddress: [color=red]array [0..MAX_ADAPTER_ADDRESS_LENGTH-1] of byte;[/color]
           PhysicalAddressLength: DWord;
           Flags: DWord;
           Mtu: DWord;
           IfType: DWord;
           OperStatus: IF_OPER_STATUS;
           Ipv6Index: DWord;
           ZoneIndices: [color=red]array [0..15] of DWORD;[/color]
           FirstPrefix: PIP_ADAPTER_PREFIX;);
      1 : (length : DWORD;
           IfIndex : DWORD;)
    end;
Nicht zu vergessen die Arrays auch in Delphi-Konvention einzubauen. (rot markiert)

Mephistopheles 1. Mai 2005 15:27

Re: C- union Struktur in Delphi
 
Zitat:

Zitat von Christian Seehase
das Problem ist, dass in der Originalstruktur die Union am Anfang steht, die kannst Du jetzt also nicht einfach an das Ende setzen.

Du darfst den varianten Teil aber auch nicht einfach an den Anfang setzen ;) ... mein Delphi verweigert sich jedenfalls dieser Syntax ;)

Siehe hier:
Zitat:

Zitat von Christian Seehase
Ein gravierendes Problem gibt es allerdings auf jeden Fall:
Der variante Teil muss am Ende der Record-Deklaration stehen, so dass man ziemliche Verrenkungen machen muss, wenn man so eine Struktur mittendrin braucht.

Zitat:

Zitat von Christian Seehase
Ich denke mal so müsste es gehen:

Code:
  PIP_ADAPTER_ADDRESSES = ^TIP_ADAPTER_ADDRESSES;
  TIP_ADAPTER_ADDRESSES = packed record
    case byte of
      0 : (Alignment : int64;
           Next : P_IP_ADAPTER_ADDRESSES;
           AdapterName: PChar;
           FirstUnicastAddress: PIP_ADAPTER_UNICAST_ADDRESS;
           FirstAnycastAddress: PIP_ADAPTER_ANYCAST_ADDRESS;
           FirstMulticastAddress: PIP_ADAPTER_MULTICAST_ADDRESS;
           FirstDnsServerAddress: PIP_ADAPTER_DNS_SERVER_ADDRESS;
           DnsSuffix: PWCHAR;
           Description: PWCHAR;
           FriendlyName: PWCHAR;
           PhysicalAddress: [color=red]array [0..MAX_ADAPTER_ADDRESS_LENGTH-1] of byte;[/color]
           PhysicalAddressLength: DWord;
           Flags: DWord;
           Mtu: DWord;
           IfType: DWord;
           OperStatus: IF_OPER_STATUS;
           Ipv6Index: DWord;
           ZoneIndices: [color=red]array [0..15] of DWORD;[/color]
           FirstPrefix: PIP_ADAPTER_PREFIX;);
      1 : (length : DWORD;
           IfIndex : DWORD;)
    end;

Definitiv nicht! Der variante Teil ist falsch plaziert.

Zitat:

Zitat von Christian Seehase
Nicht zu vergessen die Arrays auch in Delphi-Konvention einzubauen. (rot markiert)

Okay, die Arrays waren deswegen ein Problem bei mir. Hast recht. Ich beschäftige mich eindeutig zuviel mit C und C++ :zwinker:

So ist es komplett korrekt:
Code:
type
  PIP_ADAPTER_ADDRESSES = ^TIP_ADAPTER_ADDRESSES;
  TIP_ADAPTER_ADDRESSES = packed record
    case Integer of
      [color=red]1: (Alignment: ULONGLONG);
      2: (Length: ULONG;
        IfIndex: DWORD;[/color]
        [color=green]Next: PIP_ADAPTER_ADDRESSES;
        AdapterName: PChar;
        FirstUnicastAddress: PIP_ADAPTER_UNICAST_ADDRESS;
        FirstAnycastAddress: PIP_ADAPTER_ANYCAST_ADDRESS;
        FirstMulticastAddress: PIP_ADAPTER_MULTICAST_ADDRESS;
        FirstDnsServerAddress: PIP_ADAPTER_DNS_SERVER_ADDRESS;
        DnsSuffix: PWCHAR;
        Description: PWCHAR;
        FriendlyName: PWCHAR;
        PhysicalAddress: array[0..MAX_ADAPTER_ADDRESS_LENGTH - 1] of Byte;
        PhysicalAddressLength: DWORD;
        Flags: DWORD;
        Mtu: DWORD;
        IfType: DWORD;
        OperStatus: IF_OPER_STATUS;
        Ipv6Index: DWORD;
        ZoneIndices: array[0..15] of DWORD;
        FirstPrefix: PIP_ADAPTER_PREFIX;[/color]
        )
  end;

jackie 1. Mai 2005 15:32

Re: C- union Struktur in Delphi
 
Hallo!
Danke erstmal für den Code!

Im Original erstreckt sich die union Struktur ja nur über Alignment, Length und IfIndex. In eurem Code schließt die case Anweisung aber mehr als nur diese Variablen ein. Das verstehe ich nicht ganz.

Mephistopheles 1. Mai 2005 15:35

Re: C- union Struktur in Delphi
 
Zitat:

Zitat von jackie
Hallo!
Danke erstmal für den Code!

Im Original erstreckt sich die union Struktur ja nur über Alignment, Length und IfIndex. In eurem Code schließt die case Anweisung aber mehr als nur diese Variablen ein. Das verstehe ich nicht ganz.

Das ist ja gerade das Problem.

sizeof(ULONGLONG) == 8
sizeof(ULONG) + sizeof(DWORD) == 8

Die Union hat also eine Größe von 8 Byte!

Dabei überlappen sich die Member nur bis IfIndex respektive Alignment. Der Rest muß an das Ende der Union geklatscht werden, weil die Delphi-Syntax es so will ;)

jackie 1. Mai 2005 15:38

Re: C- union Struktur in Delphi
 
Danke, jetzt hab ichs endlich verstanden! :???:

Mephistopheles 1. Mai 2005 15:42

Re: C- union Struktur in Delphi
 
Habe oben nochmal farbig markiert, was Union und varianter Teil sind!

BTW: Gerade hatte ich einen Post von Christian gesehen, mit diesem Emoticon :wall: ... jetzt isser weg :shock:

Christian Seehase 1. Mai 2005 16:00

Re: C- union Struktur in Delphi
 
Moin Mephisto,

sorry, ich musste noch einmal in mich gehen ;-)

Die farbige Markierung ist allerdings, was die Delphi-Syntax angeht, nicht korrekt.
Für Delphi ist alles ab case der variante Teil.

Da für einen varianten Abschnitt ein einem Record immer soviel Speicher reserviert wird, wie für den grössten Abschnitt erforderlich ist, spielt es auch keine Rolle, ob jetzt der von Dir grün markierte Abschnitt dem ersten oder dem zweiten Abschnitt zugeschlagen wird.
Deine Variante finde ich allerdings übersichtlicher.

(ich hatte sogar mal den Fall, dass ich die Abschnitte gleich gross machen musste ;-))

Mephistopheles 1. Mai 2005 16:06

Re: C- union Struktur in Delphi
 
Zitat:

Zitat von Christian Seehase
Die farbige Markierung ist allerdings, was die Delphi-Syntax angeht, nicht korrekt.
Für Delphi ist alles ab case der variante Teil.

Mit Union wollte ich eigentlich den ursprünglichen Unionanteil anzeigen.

Zitat:

Zitat von Christian Seehase
Da für einen varianten Abschnitt ein einem Record immer soviel Speicher reserviert wird, wie für den grössten Abschnitt erforderlich ist, spielt es auch keine Rolle, ob jetzt der von Dir grün markierte Abschnitt dem ersten oder dem zweiten Abschnitt zugeschlagen wird.

Stimmt. Aber ab 2 oder 3 Unions pro record würdest du das bestimmt nicht mehr sagen :mrgreen:

Zitat:

Zitat von Christian Seehase
Deine Variante finde ich allerdings übersichtlicher.

Ab 2 oder 3 Unions dann auch nicht mehr wirklich *grins*

Robert Marquardt 1. Mai 2005 17:47

Re: C- union Struktur in Delphi
 
Man platziert den varianten Teil einfach in einem eigenen Record-Typen und kann ihn dan nan den Anfang setzen.
C ist hier etwas flexibler, da es Unions an allen Orten und auch noch unbenannt zulaesst.
Es laesst sich aber prinzipiell alles in Delphi nachmachen. Der Preis sind gelegentlich Hilfstypen und Hilfsnamen.

Assarbad 1. Mai 2005 18:43

Re: C- union Struktur in Delphi
 
Zitat:

Zitat von Robert Marquardt
Der Preis sind gelegentlich Hilfstypen und Hilfsnamen.

Also das von dir zu hören, finde ich aber schön ;) ... wo dir doch schon die 'falsche' Groß-/Kleinschreibung bei "result :=" aufstößt :zwinker:


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