Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Sonstige Fragen zu Delphi (https://www.delphipraxis.net/19-sonstige-fragen-zu-delphi/)
-   -   Aufbau Dynamisches Array? (https://www.delphipraxis.net/161286-aufbau-dynamisches-array.html)

iphi 27. Jun 2011 07:55

Aufbau Dynamisches Array?
 
Hallo,

kann mir jemand beim Verständnis des Aufbaus von dynamischen Arrays auf die Sprünge helfen?

Soweit verstehe ich die Sache:

Delphi-Quellcode:
var x: array of BYTE;
...legt einen Zeiger x an, der auf den später allokierten Speicherblock zeigen soll.

Delphi-Quellcode:
SetLength(x,4);
...allokiert 4 Byte zusammenhängenden Speicher. Auf diesen zeigt dann x.

Nun meine Frage: Wo wird denn die Länge des dynamischen Arrays im Speicher abgelegt? Ist x wirklich nur ein Zeiger oder ein Record, der die Länge enthält?

Deep-Sea 27. Jun 2011 08:06

AW: Aufbau Dynamisches Array?
 
Auszug aus der System.pas:
Delphi-Quellcode:
procedure _DynArrayLength;
asm
{       FUNCTION _DynArrayLength(const a: array of ...): Longint; }
{     ->EAX    Pointer to array or nil                          }
{     <-EAX    High bound of array + 1 or 0                      }
        TEST   EAX,EAX
        JZ     @@skip
        MOV    EAX,[EAX-4]
@@skip:
end;
Interessant ist hier nur die Zeile mit dem MOV-Befehl. Er packt in das Ausgaberegister den Wert, auf den der Array-Pointer - 4 zeigt.
Sprich: Die Länge wird direkt vor den Daten, auf die der Pointer zeigt, gespeichert.
Es handelt sich also nur um einen Pointer und keinen Record.

mleyen 27. Jun 2011 08:16

AW: Aufbau Dynamisches Array?
 
Delphi-Quellcode:
function GetArrayLen(var bytes: TBytes): Longint;
begin
  if Cardinal(bytes) = 0 then
    Result := 0
  else
    Result := PInteger(Cardinal(@bytes[0])-sizeof(Result))^;
end;
In der kommenden 64bit-Version wird da auch Int64 verwendet.

Deep-Sea 27. Jun 2011 08:19

AW: Aufbau Dynamisches Array?
 
Richtig. Mein Auszug stammt aus einer 32-bit-Version :wink:
(Aber darum ging's hier ja auch nicht wirklich.)

iphi 27. Jun 2011 08:43

AW: Aufbau Dynamisches Array?
 
Alles klar, danke!
Dann sieht der Datenblock eines dynamischen Arrays also genauso aus wie der eines statischen Arrays mit der Arraylänge ganz am Anfang, richtig?

Blup 27. Jun 2011 11:58

AW: Aufbau Dynamisches Array?
 
Nein, in einem statischen Array sind kein Informationen über die Größe enthalten.

Bei aktiver Bereichsprüfung erzeugt der Compiler lediglich Code der die Zugriffe auf den gültigen Bereich des Arrays prüft. Die Größe des Array ist dabei Teil dieses Codes.

Deep-Sea 27. Jun 2011 12:28

AW: Aufbau Dynamisches Array?
 
@Blup:
Ich glaube er wollte nur auf die Gleichheit der eigentlichen Datenbereiche hinaus. Glaube ich :gruebel:

himitsu 27. Jun 2011 13:40

AW: Aufbau Dynamisches Array?
 
Delphi-Quellcode:
// für 32 Bit-Delphi (mal sehn, ob die den StringCheckingScheiß jetzt endlich wieder ausbauen)

const
  cDynArrayInfoA  = 2 * SizeOf(LongInt);
  {$IF CompilerVersion >= 20.0}
    cDynArrayInfoS = 3 * SizeOf(LongInt);
  {$ELSE}
    cDynArrayInfoS = cDynArrayInfoA;
  {$IFEND}

type
  TDynArrayInfo = packed Record
    RefCount:    LongInt;
    ElementCount: LongInt;
    Data:        packed Array[0..High(Integer) - cDynArrayInfoA - 1] of Byte;
  End;
  PDynArrayInfo = ^TDynArrayInfo;
  // PInfo := @Var - cDynArrayInfoA;

  TAnsiStringInfo = packed Record
    {$IF CompilerVersion >= 20.0}
      CharSize:  Word; // könnte auch sein, daß CharSize und Codepage andersrum liegen
      Codepage:  Word; // (dieser Schrott stammt von dem blöden StringChecking)
    {$IFEND}
    RefCount:    LongInt;
    ElementCount: LongInt;
    Data:        packed Array[1..High(Integer) - cDynArrayInfoS] of AnsiChar;
  End;
  PAnsiStringInfo = ^TAnsiStringInfo;
  // PInfo := @Var - cDynArrayInfoS;

  TUnicodeStringInfo = packed Record
    {$IF CompilerVersion >= 20.0}
      CharSize:  Word;
      Codepage:  Word;
    {$IFEND}
    RefCount:    LongInt;
    ElementCount: LongInt;
    Data:        packed Array[1..(High(Integer) - cDynArrayInfoS) div 2] of WideChar;
  End;
  PUnicodeStringInfo = ^TUnicodeStringInfo;
  // PInfo := @Var - cDynArrayInfoS;

  TDynArrayInfo<TheField> = packed Record
    RefCount:    LongInt;
    ElementCount: LongInt;
    Data:        packed Array[0..(High(Integer) - cDynArrayInfoA) div SizeOf(TheField) - 1] of TheField;
  End;
  TStringInfo<TheChar> = packed Record
    {$IF CompilerVersion >= 20.0}
      CharSize:  Word;
      Codepage:  Word;
    {$IFEND}
    RefCount:    LongInt;
    ElementCount: LongInt;
    Data:        packed Array[1..(High(Integer) - cDynArrayInfoS) div SizeOf(TheChar)] of TheChar;
  End;
Der interne Zeiger zeigt jeweils auf .Data

Delphi-Quellcode:
var
  MyArray = array of irgendwas;

if MyArray <> nil then
  TheLength := PDynArrayInfo(Integer(MyArray) - SDynArrayInfoA).ElementCount
else
  TheLength := 0;
PS: Bei den Strings ist jeweils noch ein abschließendes #0 nach den Stringdaten, für die Kompatibilität und eine direkte Konvertierungsmöglichkeit in einen PChar/PAnsiChar/PWideChar.
Es ist also ein DynArray, mit einem zusätzlichen Feld.


Und wenn man sich nun den "alten" Delphi-Referenz durchsuchenShortString ansieht,
dann erkennt man, warum Delphi-Strings mit 1 beginnen.

Delphi-Quellcode:
type
  TShortStringInfo = packed Record
    Len: Byte;
    Chars: packed Array[1..255{1..Len}] of Char;
  End;
  PShortStringInfo = ^TShortStringInfo;
  // PInfo := @Var;


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