![]() |
Problem bei FreeMem
Morgen.
Im IRC gabs grad ne kleine Diskussion um das löschen des ersten Zeichens eines Strings. Ich hatte folgende Idee:
Delphi-Quellcode:
Tja, gibt nur leider ne ungültige Pointeroperation bei freemem.
procedure DeleteFirstChar(var str: String);
var addr: Pointer; begin addr:=@str[1]; str:=PChar(@str[2]); freemem(addr, sizeof(Char)); end; Warum? Ich kann per addr^ noch auf den Char zugreifen, aber seltsamerweise nicht löschen. |
Re: Problem bei FreeMem
1. Wie kommst du auf die Idee, dass der String mit AllocMem() alloziiert wurde so dass du ihn per FreeMem() wieder freigeben könntest?
2. Wieso sollte der Char im Speicher verschwinden? So lange nix neues an dessen Stelle geschrieben wird, bleibt sein Inhalt. Es geht doch kein Wiper Process drüber und setzt irgendwelche Werte in den Speicherbereichen die freigegeben wurden. Vor allem wäre dann die Frage: welcher Inhalt würde bedeutet dass der Speicher leer ist? $44 oder $00? Oder vielleicht doch eher ein Inhalt von $ff ? |
Re: Problem bei FreeMem
:gruebel:
Wie bitte? Wenn ich mache str:=PChar(@str[2]); dann habe ich das erste Zeichen entfernt. Es hieß aber im IRC, dann hätte ich ein Memory-Leak, weil das Zeichen, das nicht mehr im String ist, später nicht freigegeben wird. |
Re: Problem bei FreeMem
Abgesehen von deinem Problem ...
was spricht gegen folgendes:
Delphi-Quellcode:
procedure DeleteFirstChar(var S: String);
begin S := Copy(S, 2, Length(S) - 1); end; |
Re: Problem bei FreeMem
Ein String wird zwar er GetMem reserviert, allerdings geschieht da noch mehr.
so geht es ... aber mein Tipp ... schau dir unbedingt mal an, was wirklich für Code ausgeführt wird :zwinker: (im CPU-Fenster)
Delphi-Quellcode:
Ein String ist wie jedes dynamische Array voll von CompilerMagic und "versteckten" Funktionen ^^
procedure DeleteFirstChar(var str: String);
begin if Length(str) >= 2 then str := PChar(@str[2]); end; Ach ja ... du hattes 'ne Prüfung vergessen ... str[2] ergibt ja 'nen netten Fehler, wenn es kein 2. Zeichen gibt :warn: ich weiß grad nicht wie weit die CodeOptimierung geht, aber vermutlich wird das noch ä bissl schneller sein, wenn der String seinen Speicher geteilt hat.
Delphi-Quellcode:
Wenn du es am Schnellsten haben willst, dann so ... schneller geht es per Pascal nicht
procedure DeleteFirstChar(var str: String);
begin if str <> '' then str := PChar(PInteger(str) + 1); end; (nicht getestet, aber theoretisch korrekt ... hoff ich mal ._.)
Delphi-Quellcode:
Zum Verstehen dieser Methode empfehle ich ein Studium der Arbeitsweise des der AnsiStrings und eventuell auch noch ein Grundverständnis für MemoryManager. :angel:
procedure DeleteFirstChar(var str: String);
begin if Str = '' then Exit; UniqueString(Str); Move(PLongInt(Str) + 1, PLongInt(Str), (PLongInt(Str) - 4)^); Dec((PLongInt(Str) - 4)^); end; |
Re: Problem bei FreeMem
Zitat:
|
Re: Problem bei FreeMem
Liste der Anhänge anzeigen (Anzahl: 1)
Im grunde läßt meine Methode auch einige Bytes mehr im Speicher, als nötig, aber diese werden nachher automatisch freigegeben.
und du kannst kein einzelnes Char freigeben, denn es ist ein zusammenhängender Speicherblock ... es ist also unmöglich nur ein einzelnes Byte freizugeben. Man könnte allerhöchstens den gesamten Block (also alles) freigeben
Delphi-Quellcode:
is natürlich schneller als str := ''; , denn hier wird ja der Speicher direkt freigegeben und nicht erst über Umwege.
FreeMem(PInteger(str) - 8);
Schau mal in den Anhang ... Abschnitt "Compiler Intern Data-Types", dort findest du unteranderem die interne Struktur der Delphi-Strings. Ach ja: die Copy-Variante ist schneller als deine, denn diese ließt einfach die Stringlänge aus und muß nicht erst die Länge ermitteln. Außerdem ist Copy dahingehend optimiert, daß eventuell nur innherlb eines Speicherblocks kopiert wird ... bei dir wird immer ein 2. Block angelgt und danach (wenn nötig) der OriginalString freigegeben. |
Re: Problem bei FreeMem
Stimmt, meine Methode ist um den Faktor 5/4 langsamer...
Also ich hätte mir das einfach so vorgestellt, dass ich den Pointer einfach ein Zeichen weiter nach hinten verschiebe. Geht anscheinend nicht... |
Re: Problem bei FreeMem
Nein, natürlich nicht, denn der Pointer ist fest und zeigt sozusagen mehr auf die Stringstruktur und nicht auf die Daten ... es müssen also schon die Daten verschoben werden.
|
Re: Problem bei FreeMem
Warum eigentlich? Warum kann man nicht den Pointer einfach um eins inkrementieren? :gruebel:
|
Re: Problem bei FreeMem
Weil der String halt nicht so arbeitet :zwinker:
str ist ja "nur" ein pointer auf die Stringstruktur
Code:
der Zeiger in str zeigt zwar nicht wie allgemein üblich auf den Anfang des resservierten Datenbereichs, sonden aud das erste Zeichen,
RRRRCCCCDDDDDDDDDDDDDDDDDDDDDDD.....#0
^ ^ ^ ^ ^ ^ ab hier sind dann alle Elemente (Zeichen) aufgereit ^ Elemete im Array ... entspricht beim String der Zeichenanzahl Refferenzzähler ^ ^ ^ ^____________________________________^ diese Daten liegen [b]einem[/b] Speicherblock dies ist allerdings nur weil es halt so einfacher ist den string nach PChar zu casten. zur Speicherverwaltung wird demnach immer PInteger(str)-2*SizeOf(Integer) an GetMem, ReallocMem und FreeMem geleitet. Und da gibt es Probleme, sobald du den String mal verändernfreigeben willst :warn: Length arbeitet ja auch nur mit PInteger(PInteger(str)-SizeOf(Integer))^, was auch falsche werte liefern würd, wenn dman daran rumfummelt. |
Re: Problem bei FreeMem
Und die Funktion Delete() im Gegensatz dazu? Wie schnell/langsam ist diese dazu?
|
Re: Problem bei FreeMem
Jenachdem wie der MemoryManager arbeitet und wie lang gerade der String ist, ist meine "schnelle" Version vielleicht 1/16 bis 1/4 aller Fälle schneller ... bei mir wird ja der Speicherblock nicht gekürtzt und demnach entfällt eine eventuelle Änderung, was bis hin zum Kopieren des gesamten Strings gehen könnte.
Aber Delete ist meistens schonmal schneller als Copy, da es den String direkt bearbeitet. Copy legt ja immer erst 'ne Kopie an, welche per Result übergeben wird und erst dann wird das Original durch die Kopie ersetzt. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 03:24 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