![]() |
Cast Array[0..x] of Byte in TBytes
Hi!
Ich möchte gerne mit TIdTCPServer Daten an den Client schicken. Über
Delphi-Quellcode:
Kann ich TBytes an den Client schicken.
AContext.Connection.IOHandler.Write(Buffer, len);
Ich habe aber nun Buffer als Array[0..1023] of Byte; Nun will Write den Buffer nicht mehr. Und extra immer ein TBytes Array mit Setlength zu setzen, dann die Daten kopieren und diese dann zu schicken ist etwas umständlich. Wie kann man das richtig Casten damit es keine Probleme gibt? |
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
|
AW: Cast Array[0..x] of Byte in TBytes
Ok, danke!
So gibt es zumindest keinen Crash:
Delphi-Quellcode:
Man spart sich das Memcopy.
var
_buffer : TBytes; buffer : Array[0..1023] of Byte; begin SetLength(_buffer, Length(buffer)); _buffer := @buffer[0]; |
AW: Cast Array[0..x] of Byte in TBytes
Das kann nicht richtig sein.
Du verlierst den Zeiger auf den dynamisch reservierten Speicher (_buffer). Es sollte ein Speicherleck geben und fast eine Zugriffsverletzung, wenn _buffer irgendwann freigegeben werden soll und buffer kein gültiger Zeiger mehr ist. |
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
|
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
Durch den Umstand, daß die Write-Methode den Parameter ABuffer als Konstante erwartet, wird vom Compiler nur die Anfangsadresse des Arrays übergeben. Und damit ist es völlig egal, ob es sich dabei um den Beginn eines dynamischen oder eines statischen Arrays handelt. Ein kleines Beispiel:
Delphi-Quellcode:
Das heißt: Ein Typecast ist sehr wohl möglich. Ob er auch sinnvoll ist, steht auf einem ganz anderen Blatt... :wink:
procedure Test(const ABuffer: TBytes; ALength: Integer);
var n: Integer; s: String; begin s:=''; for n:=0 to ALength-1 do begin if s<>'' then s:=s+','; s:=s+IntToStr(ABuffer[n]); end; ShowMessage(s); end; procedure TForm1.Button1Click(Sender: TObject); var n: Integer; a: array[0..9] of Byte; b: TBytes; begin //dynamisches Array: SetLength(b,10); for n:=Low(b) to High(b) do b[n]:=n; Test(b,10); //statisches Array: for n:=Low(a) to High(a) do a[n]:=n; Test(TBytes(@a[0]),10); //<- Typecast end; |
AW: Cast Array[0..x] of Byte in TBytes
Nur zum Beruhigen: ich habe es so eh nicht umgesetzt!
Bei meinen Versuchen war das die einzige Methode wo es keinen Memory Fehler gab :oops: |
AW: Cast Array[0..x] of Byte in TBytes
@schwa266: Sowas hat jeder mal versucht, besonders wenn man die Verwandschaft (und Compiler-Unterstützung) von String und PChar kennt, will man das gerne mal auf andere Bereiche übertragen.
Zitat:
Delphi-Quellcode:
"From 0 To <Random>"
procedure Test(ABuffer: TBytes; ALength: Integer);
var n: Integer; s: String; begin ShowMessage(Format('From %d To %d', [Low(ABuffer), High(ABuffer)])); Wie das Beispiel zeigt ist der Typecast nicht korrekt und nur im begrenzten Fall problemlos. |
AW: Cast Array[0..x] of Byte in TBytes
@Satty67:
Ich hatte auf die Behauptung von Uwe Raabe reagiert - nicht auf Deine. Und es ging bei der Frage des Threaderstellers darum, ob es in dem konkteren Fall möglich ist. Noch einmal ausführlich für Dich, was ich mit meiner Bemerkung gemeint hatte: Selbstverständlich ist es eine bekannte Tatsache, daß ein Pointer ein Pointer bleibt und der Typecast nur ein (unsauberes) Mittel ist, um den Compiler dazu zu bringen, seine Arbeit ohne Fehlermitteilung fortzusetzen. Man muß also sehr genau wissen, was man tut und sich überlegen, welche Delphi-Funktionen mit dem gecasteten Pointer eingesetzt werden können. Genau aus diesem Grund hatte ich geschrieben, daß man sich die Frage stellen sollte, ob ein Cast sinnvoll ist - also ob er das gewünschte Ergebnis liefert. Geht es zum Beispiel um eine hohe Geschwindigkeit, wäre das ständige Kopieren von Puffern kontraproduktiv. Natürlich wäre es günstiger - falls möglich - , das statische Array gleich durch ein dynamisches Array zu ersetzten und damit potentiellen Problemen aus dem Weg zu gehen. Hatte ich an irgend einer Stelle behauptet, daß ein derartiger Cast immer und in jedem Fall sinnvoll ist? Ich kann mich nicht daran erinnern. Es ist also von Dir absolut überflüssig, in einer derart belehrenden Art und Weise zu schreiben und andere für dumm zu halten. Aber das scheint hier im Forum leider eine weit verbreitete Unsitte zu sein. Da wird nicht auf konkrete Fragen geantwortet sondern lieber aus - nahezu - jedem Thread eine Prizipiendiskussion gemacht, um sich selbst zu profilieren. Sehr bezeichnend und sehr bedauerlich. |
AW: Cast Array[0..x] of Byte in TBytes
Liste der Anhänge anzeigen (Anzahl: 1)
Zitat:
Das tatsächliche Speicherlayout kann man ![]() Beiliegendes Projekt zeigt das ganz deutlich - und es crasht auch beim Beenden. Insofern bleibe ich bei meiner Aussage, daß das Speicherlayout unterschiedlich ist. |
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
Delphi-Quellcode:
nach
TBytes
Delphi-Quellcode:
läßt sich Casten, da die nötigen Daten des statischen arrays im dynamischen enthalten sind, solange das dynamische Array mindestens so groß ist, wie das Statische.
Array[0..x] of Byte
Aber in diesem Fall kommt man um ein Umkopieren nicht drumrum. |
AW: Cast Array[0..x] of Byte in TBytes
Der TIdIOHandler nimmt aber auch u.a. Streams an. Ich habe mir vor einiger Zeit schon eine kleine Klasse von TCustomMemoryStream abgeleitet, der ich im Konstruktor einen bereits bestehenden Puffer - wie z.B. ein statisches Array - und dessen Länge übergeben kann. So wandelt man irgendwelche Daten ohne umkopieren in einen Stream :stupid:
Ob es hier aber nicht vlt. einfacher und sinnvoller wäre, den aufrufenden Code an TBytes anzupassen, kann ich nicht beurteilen. |
AW: Cast Array[0..x] of Byte in TBytes
@Uwe Raabe:
Die Nutzdaten haben das selbe Layout: Einfach 1024 Bytes hintereinander. Die Verwaltungsinformationen (auf die Delphi intern zugreift) sind für diese Speicherbereiche unterschiedlich. Und im konkteten Fall der Write-Methode sind nur die Nutzdaten relevant. Genau aus diesem Grund muß ja auch die Länge des Speicherbereichs mit übergeben werden. Und damit ist ein Typecast im Gegensatz zu Deiner Behauptung im konkreten Falle sehr wohl möglich. Sorry - aber es wie im Kindergarten... :lol: Jeder will hier auf Biegen oder Brechen Recht behalten. Ist irgendwie lächerlich und schade um die vergeudete Zeit.. :stupid: Und wie überflüssig das Ganze ist, sieht man am Beitrag des Fragestellers: Es hat bei ihm (im speziellen Fall) funktioniert. Auch bei
Delphi-Quellcode:
treten (bei mir) keine Speicherfehler auf.
ReportMemoryLeaksOnShutdown:=true;
Aber die Theoretiker beharren auf ihrer Meinung... :stupid: @Deep-Sea: Interessante Idee! :thumb: |
AW: Cast Array[0..x] of Byte in TBytes
Auch wenn ich mich ungern in die Schussbahn einer so hitzigen Diskussion begebe, wollte ich es trotzdem los werden:
Zitat:
Delphi-Quellcode:
sieht so aus:
AContext.Connection.IOHandler.Write
Delphi-Quellcode:
.
LLength := IndyLength(ABuffer, ALength, AOffset);
Und die zweite Zeile der Funktion IndyLength enthält wiederum das:
Delphi-Quellcode:
Hier wird also auf die Länge des dyn. Arrays mittels Length zugegriffen. Das das gut geht ist reine Glückssache ...
LAvailable := IndyMax(Length(ABuffer)-AIndex, 0);
Dein Code funktioniert also, sofern man die Funktion selbst geschrieben hat und auf Length verzichtet. Allgemein ist es aber leider eine tickende Zeitbombe ... |
AW: Cast Array[0..x] of Byte in TBytes
PS: Ist denn unbedingt ein statisches Array notwendig?
Einfach TBytes nehmen und am Anfang ein
Delphi-Quellcode:
und schon kann man das Array direkt übergeben.
SetLength(a, 1000);
Wenn irgendwo, in deinem Code, sowas wie @a vorkommt, dann dieses durch @a[0] ersetzen und das Selbe auch noch bei Var-Parametern. Zitat:
|
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
Delphi-Quellcode:
schreiben, dann passt es immer, auch für statische Arrays die nicht mit 0 beginnen :stupid:
@a[Low(a)]
|
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
Deshalb ja auch meine Bemerkung Zitat:
|
AW: Cast Array[0..x] of Byte in TBytes
Zitat:
|
AW: Cast Array[0..x] of Byte in TBytes
Ich weiß nicht, wie alt Du bist und ob Du viel mit andere Menschen zu tun hat (Kinder, Schüler, Praktikanten, Angestellte,...).
Ist aber eigentlich auch egal: Aus meiner Sicht ist oftmals der Ton wichtiger als der Inhalt. Und genau dieser Umgangston läßt in vielen Foren arg zu wünschen übrig... Aber das ist Offtopic: Wenn Du Interesse hat, können wir das auch gern per PM besprechen. |
AW: Cast Array[0..x] of Byte in TBytes
Hallo....
nu kommt doch bitte auf den Boden zurück. Können wir uns darauf einigen, daß Typecasting ein ganz heißes Eisen ist, und daß man damit sehr zurückhaltend und vorsichtig umgehen sollte? (und Pointer gehören in die gleiche Kathegorie?) Gruß K-H |
AW: Cast Array[0..x] of Byte in TBytes
Kein Problem damit. :lol:
Genau das sage ich schon seit x Beiträgen: Es ist möglich, aber nicht immer sinnvoll... (ist jetzt bestimmt schon das vierte oder fünfte mal). Und damit schließe ich meine Senftube, damit der Thread nicht durch einen Moderator geschlossen wird. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 10:21 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