![]() |
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:
...legt einen Zeiger x an, der auf den später allokierten Speicherblock zeigen soll.
var x: array of BYTE;
Delphi-Quellcode:
...allokiert 4 Byte zusammenhängenden Speicher. Auf diesen zeigt dann x.
SetLength(x,4);
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? |
AW: Aufbau Dynamisches Array?
Auszug aus der System.pas:
Delphi-Quellcode:
Interessant ist hier nur die Zeile mit dem MOV-Befehl. Er packt in das Ausgaberegister den Wert, auf den der Array-Pointer - 4 zeigt.
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; 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. |
AW: Aufbau Dynamisches Array?
Delphi-Quellcode:
In der kommenden 64bit-Version wird da auch Int64 verwendet.
function GetArrayLen(var bytes: TBytes): Longint;
begin if Cardinal(bytes) = 0 then Result := 0 else Result := PInteger(Cardinal(@bytes[0])-sizeof(Result))^; end; |
AW: Aufbau Dynamisches Array?
Richtig. Mein Auszug stammt aus einer 32-bit-Version :wink:
(Aber darum ging's hier ja auch nicht wirklich.) |
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? |
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. |
AW: Aufbau Dynamisches Array?
@Blup:
Ich glaube er wollte nur auf die Gleichheit der eigentlichen Datenbereiche hinaus. Glaube ich :gruebel: |
AW: Aufbau Dynamisches Array?
Delphi-Quellcode:
Der interne Zeiger zeigt jeweils auf .Data
// 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;
Delphi-Quellcode:
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.
var
MyArray = array of irgendwas; if MyArray <> nil then TheLength := PDynArrayInfo(Integer(MyArray) - SDynArrayInfoA).ElementCount else TheLength := 0; Es ist also ein DynArray, mit einem zusätzlichen Feld. Und wenn man sich nun den "alten" ![]() 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 09:19 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