Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Multimedia (https://www.delphipraxis.net/16-multimedia/)
-   -   Delphi frage zu strings+Char (https://www.delphipraxis.net/7120-frage-zu-strings-char.html)

iLLe 2. Aug 2003 15:29


frage zu strings+Char
 
mal was ganz einfaches :
Code:
var s:string;
begin
s:='hi';
Messagebox(0,@s[1],'info',64)
end;
undzwar ist meine frage der teil
Code:
,@s[1],
von der Messagebox.Der Teil muss ja Char sein aber ich hab einen string und was bewirkt das @ wird das dann umgewandelt oder wie ? und wenn ich [1] hinter dem s weglasse kommen komische zeichen in der messagebox was hat das für ne funktion ?

Christian Seehase 2. Aug 2003 15:44

Re: frage zu strings+Char
 
Moin iLLe,

an der von Dir genannte Stelle erwartet die Funktion die Adresse eines nullterminierten Strings (Typ PChar), da in C Strings i.d.R. intern keine Längenangabe haben, und das Ende eines Strings mit einer binären Null festgestellt wird.

Mit dem s[1] wird das erste erste Zeichen des Strings angesprochen, und das @ davor bedeutet, dass der Compiler die Adresse nehmen soll. Gesamt
heisst das für den Compiler:
Trage die Adresse des ersten Zeichens von s ein.

Die Adresse eines Zeichens lässt sich auch als PChar bezeichnen.

Eine Stringvariable enthält eigentlich "nur" die Adresse des Strings, so dass @s den Pointer auf die Adresse zurückgibt, und nicht auf den Inhalt, also die Adresse eines Pointers!

Das diese Schreibweise (@s[1]) funktioniert liegt übrigens nur daran, dass Borland Stringvariablen intern automatisch nullterminiert. Wäre das nicht der Fall hättest Du mit dieser Variante ein Problem ;-)

Die, üblicher Weise verwendeten, Hugestrings bestehen aus einem Referenzzähler (integer), einem Längenfeld (integer), dem Inhalt (char) und einer binäre Null (char), wobei (in Deinem Beispiel) s die Adresse des Inhaltes enthält. Da der Referenzzähler und das Längenfeld vor dem Inhalt liegen, beginnt der Referenzzähler auf der Adresse s-8, und das Längenfeld auf der Adresse s-4. Die Null liegt dann auf s+Inhalt Längenfeld.

iLLe 2. Aug 2003 15:51

Re: frage zu strings+Char
 
Ich weiss zwar nicht was nullterminierte strings sind und dden letzten abschnitt hab ich auch solala verstanden aber den rest weiss ich jetzt danke :)

Duffy 2. Aug 2003 15:51

Re: frage zu strings+Char
 
Hallo iLLe,
in Ergänzung und weiteren Erläuterung zum Beitrag von Christian Seehase

die Funktion MessageBox erwartet als zweiten Parameter einen Zeiger auf den String. Nach guter alter Manier, hat Delphi aber im Byte Null des Strings das Längenbyte stehen, damit ist Delphi auch kompatibel zu Kurz Strings. Siehe nachfolgende Erläuterung. Damit aber der String korrekt ausgegeben wird, sagst Du einfach "Bitte den String "S" and der Addresse "@" + 1 Byte [1] ausgeben.
Zitat:

Delphi-Sprachreferenz - Kurze String-Typen
Ein ShortString hat eine Länge von 0 bis 255 Zeichen. Obwohl sich seine Länge dynamisch ändern kann, beträgt die statische Speicherplatzzuweisung immer 256 Bytes. Im ersten Byte wird die Länge des Strings gespeichert, die restlichen 255 Bytes stehen für die Zeichen zur Verfügung. Wenn S eine ShortString-Variable ist, liefert Ord(S[0]) die Länge von S zurück (dasselbe Ergebnis erzielen Sie mit Length(S)). Durch Zuweisung eines Wertes an S[0] können Sie (wie durch einen Aufruf von SetLength) die Länge von S ändern. ShortString wird nur aus Gründen der Abwärtskompatibilität mitgeführt.

Delphi unterstützt kurze String-Typen (Untertypen von ShortString), deren maximale Länge zwischen 0 und 255 Zeichen liegen kann. Diese Typen werden mit einer Zahl in eckigen Klammern dargestellt, die auf das reservierte Wort string folgt. Ein Beispiel:

var MyString: string[100];

Hier wird die Variable MyString mit einer maximalen Länge von 100 Zeichen erstellt. Die folgenden Deklarationen sind mit der obigen Zeile identisch:

type CString = string[100];
var MyString: CString;

Bei Variablen, die auf diese Weise deklariert werden, wird dem Typ nur so viel Speicherplatz zugewiesen, wie für die angegeben Länge plus ein Byte erforderlich ist. Im obigen Beispiel belegt MyString 101 Bytes. Für eine Variable des vordefinierten Typs ShortString wären dagegen 256 Bytes erforderlich.

Bei einer Wertzuweisung an eine kurze String-Variable wird der String abgeschnitten, wenn die maximale Länge für den Typ überschritten wird.

Die Standardfunktionen High und Low bearbeiten Variablen und Typbezeichner für kurze Strings. High liefert die maximale Länge des kurzen String-Typs, während Low Null zurückgibt.
bye

Duffy 2. Aug 2003 15:53

Re: frage zu strings+Char
 
Hallo iLLe,
bei Null terminierten Strings wird das Ende eines Strings durch eine binäre Null (0) erkannt.
bye

iLLe 2. Aug 2003 15:59

Re: frage zu strings+Char
 
danke :bouncing4:

Christian Seehase 2. Aug 2003 16:16

Re: frage zu strings+Char
 
Moin Duffy,

Zitat:

Zitat von Duffy
Nach guter alter Manier, hat Delphi aber im Byte Null des Strings das Längenbyte stehen, damit ist Delphi auch kompatibel zu Kurz Strings.

Nein, denn bei Hugestrings ist das Längenfeld ein integer (4 Byte), und liegt vor dem String.
An der Stelle, an der ein HugeString das erste Zeichen enthält, befindet sich bei ShortStrings das Längenbyte.
Ausserdem werden ShortStrings nicht automatisch null terminiert.

Chewie 2. Aug 2003 16:27

Re: frage zu strings+Char
 
Zitat:

Zitat von Christian Seehase
beginnt der Referenzzähler auf der Adresse s-8, und das Längenfeld auf der Adresse s-4.

Müsste es nicht s-7 bzw. s-3 sein? Denn der eigentliche Inhalt beginnt ja am Index 1, nicht 0, also wäre die 0 nicht belegt.
Oder täusche ich mich da?

Duffy 2. Aug 2003 16:31

Re: frage zu strings+Char
 
Hallo Christian Seehase,
Zitat:

Zitat von Christian Seehase
Moin Duffy,

Zitat:

Zitat von Duffy
Nach guter alter Manier, hat Delphi aber im Byte Null des Strings das Längenbyte stehen, damit ist Delphi auch kompatibel zu Kurz Strings.

Nein, denn bei Hugestrings ist das Längenfeld ein integer (4 Byte), und liegt vor dem String.
An der Stelle, an der ein HugeString das erste Zeichen enthält, befindet sich bei ShortStrings das Längenbyte.
Ausserdem werden ShortStrings nicht automatisch null terminiert.

ist mir bekannt, halte ich aber für Erläuterung für nicht interessant, da sich das Displacement nicht ändern würde.
bye

Christian Seehase 2. Aug 2003 17:33

Re: frage zu strings+Char
 
Moin Chewie,

nein, denn der Inhalt beginnt ja bei Offset 0, so dass s+0 = Adresse des ersten Zeichens des Inhaltes.
Du hast hier offensichtlich die Indizes und die Offsets verwechselt ;-)

@Duffy:
Ich wollte eigentlich darauf hinaus, dass Delphi bei HugeStrings eben nicht "nach guter alter Manier" ein Längenbyte verwendet.
Das gilt nur für die ShortStrings.

Das folgende kann man mit einem HugeString gar nicht machen,

Delphi-Quellcode:
var
  sShort : string[10];

begin
  sShort := '01234';
  ShowMessage(IntToStr(ord(sShort[0])));
end;
da der Compiler das schon nicht zulässt.


Bei HugeStrings müsste man das so machen:

Delphi-Quellcode:
var
  sHuge : string;

begin
  sHuge := '01234';
  ShowMessage(IntToStr(PInteger(PChar(sHuge)-4)^));
end;
um direkt an den Längenzähler heranzukommen.

Das wiederum liesse sich bei ShortStrings so ähnlich machen

Delphi-Quellcode:
var
  sShort : string[10];

begin
  sShort := '01234';
  ShowMessage(IntToStr(PByte(PChar(@sShort))^));
end;

Chewie 2. Aug 2003 17:43

Re: frage zu strings+Char
 
Zitat:

Zitat von Christian Seehase
nein, denn der Inhalt beginnt ja bei Offset 0, so dass s+0 = Adresse des ersten Zeichens des Inhaltes.
Du hast hier offensichtlich die Indizes und die Offsets verwechselt ;-)

:shock: Ich glaub, ich stehe etwas auf dem Schlauch, aber was ist in diesem Fall ein Offset und was ein Index? :?

Christian Seehase 2. Aug 2003 18:29

Re: frage zu strings+Char
 
Moin Chewie,

angenommen sHuge sei eine Variable vom Typ string, dann können die einzelnen Zeichen über den Index erreicht werden, wobei der Index von 1 bis length(sHuge) reicht.
sHuge enthält die Adresse des ersten Zeichens des Inhaltes, also wäre der Offset des ersten Zeichens, relativ zur Adresse des ersten Zeichens gleich 0.

Pseudemys Nelsoni 2. Aug 2003 19:20

Re: frage zu strings+Char
 
Zitat:

Zitat von Christian Seehase
Moin Chewie,

nein, denn der Inhalt beginnt ja bei Offset 0, so dass s+0 = Adresse des ersten Zeichens des Inhaltes.
Du hast hier offensichtlich die Indizes und die Offsets verwechselt ;-)

@Duffy:
Ich wollte eigentlich darauf hinaus, dass Delphi bei HugeStrings eben nicht "nach guter alter Manier" ein Längenbyte verwendet.
Das gilt nur für die ShortStrings.

Das folgende kann man mit einem HugeString gar nicht machen,

Delphi-Quellcode:
var
  sShort : string[10];

begin
  sShort := '01234';
  ShowMessage(IntToStr(ord(sShort[0])));
end;
da der Compiler das schon nicht zulässt.


Bei HugeStrings müsste man das so machen:

Delphi-Quellcode:
var
  sHuge : string;

begin
  sHuge := '01234';
  ShowMessage(IntToStr(PInteger(PChar(sHuge)-4)^));
end;
um direkt an den Längenzähler heranzukommen.

Das wiederum liesse sich bei ShortStrings so ähnlich machen

Delphi-Quellcode:
var
  sShort : string[10];

begin
  sShort := '01234';
  ShowMessage(IntToStr(PByte(PChar(@sShort))^));
end;

was bedeutet -4? habs mal durch -5 oder -3 ersetzt da kommen ja 341234523 zahlen raus

Duffy 2. Aug 2003 21:53

Re: frage zu strings+Char
 
Hallo Christian Seehase,
MyString[1] ist bei einem Short String und einem Huge String ein und dasselbe Zeichen. Das ist eben so, egal ob mit einem Längenbyte oder mit 4.
bye

Christian Seehase 3. Aug 2003 00:10

Re: frage zu strings+Char
 
Moin Zusammen,

@Silent:
Bei einem HugeString befindet sich auf den Bytes -8 bis -5 der Referenzzähler, und auf -4 bis -1 das Längenfeld des Strings (Werte relativ zu der Adresse in sHuge)
Deshalb -4, da dann auf das niederwertigste Byte des Längenfeldes adressiert wird.

Offsets relativ zu der Adresse in sHuge:
(Wert darunter in Hex, bezogen auf das Beispiel)

Code:
Referenzz. |Länge       | Inhalt
-8 -7 -6 -5 | -4 -3 -2 -1 |  0  1  2  3  4  5  6
FF FF FF FF | 05 00 00 00 | 30 31 32 33 34 35 00
Hier hat der Referenzzähler den Wert -1, da sHuge direkt eine Stringkonstante zugewiesen wurde (sHuge := '01234'). Diese haben immer den Referenzzähler -1, da sie ihren Wert nicht verändern.
Wichtig ist hierbei noch, dass die bei Intel übliche Speicherreihenfolge Verwendung findet, also das niederwertigste Byte zuerst, das höchstwertigst zuletzt. Deshalb steht die 5 auf Offset -4.
Würdest Du also bei meinem Beispiel -5 statt -4 schreiben, müsste als Ergebnis 1535 (dezimal) herauskommen.

Code:
-5 -4 -3 -2
FF 05 00 00 => 000005FF Hex => 1535 Dez
und bei -3

Code:
-3 -2 -1  0
00 00 00 30 => 30000000 Hex => 805306368 Dez
@Duffy:
Das ist natürlich richtig, aber ich finde, dass es nicht schaden kann mal auf die Unterschiede der Stringtypen einzugehen.

Duffy 3. Aug 2003 00:12

Re: frage zu strings+Char
 
Hallo Christian Seehase,
schaden kann es auf keinen Fall ...
bye

Pseudemys Nelsoni 3. Aug 2003 02:02

Re: frage zu strings+Char
 
danke für die ausführliche erklärung chris


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