Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Win32/Win64 API (native code) (https://www.delphipraxis.net/17-win32-win64-api-native-code/)
-   -   Delphi Buffer bei API Strukturlisten - reicht Freemem(Buffer) aus? (https://www.delphipraxis.net/200610-buffer-bei-api-strukturlisten-reicht-freemem-buffer-aus.html)

BitSum 7. Mai 2019 11:47

Buffer bei API Strukturlisten - reicht Freemem(Buffer) aus?
 
Hallo,
ich habe eine generelle Frage zu enum Funktionen der Windows API.

Man muss ja i.d.R. Speicher für ein Strukturelement per pElement := AllocMem(Buffersize) reservieren.

aber reicht es dann einfach am Ende den Speicher per FreeMem(pElement) freizugeben damit die ganze Liste freigegeben wird oder muss ich mich um die Freigabe jedes Listenelements selber kümmern?

Rudy Velthuis 7. Mai 2019 13:00

AW: Buffer bei API Strukturlisten - reicht Freemem(Buffer) aus?
 
Zitat:

Zitat von BitSum (Beitrag 1431731)
Hallo,
ich habe eine generelle Frage zu enum Funktionen der Windows API.

Man muss ja i.d.R. Speicher für ein Strukturelement per pElement := AllocMem(Buffersize) reservieren.

aber reicht es dann einfach am Ende den Speicher per FreeMem(pElement) freizugeben damit die ganze Liste freigegeben wird oder muss ich mich um die Freigabe jedes Listenelements selber kümmern?

Ich würde da lieber dynamische Arrays benutzen, in der heutigen Zeit in der Form TArray<BasisTyp>.

Genauso würde ich für Records (Structs) lieber New und Dispose benutzen als GetMem/AllocMem und FreeMem.

Normalerweise muss man, wenn es um APIs geht, nicht jedes Element freigeben, es sei denn, das sind Zeiger auf Strukturen die gesondert mit New oder GetMem alloziert wurden.

peterbelow 7. Mai 2019 16:11

AW: Buffer bei API Strukturlisten - reicht Freemem(Buffer) aus?
 
Zitat:

Zitat von BitSum (Beitrag 1431731)
Hallo,
ich habe eine generelle Frage zu enum Funktionen der Windows API.

Man muss ja i.d.R. Speicher für ein Strukturelement per pElement := AllocMem(Buffersize) reservieren.

aber reicht es dann einfach am Ende den Speicher per FreeMem(pElement) freizugeben damit die ganze Liste freigegeben wird oder muss ich mich um die Freigabe jedes Listenelements selber kümmern?

Von was für "Listenelementen" redest Du hier? Selbst wenn die API-Funktion einen ganzen Array von Elementen in deinem Speicherblock ablegt ist es immer noch ein einziger Speicherblock, also reicht ein FreeMem aus, egal was der Block enthält. Du mußt nur aufpassen, dass Du einen Zeiger auf den Anfang des Blocks an FreeMem übergibt, den von AllocMem zurückgegebenen Pointer also nicht irgendwie überschreibst, z. b. wenn Du dich durch die Elemente im Block arbeitest.

Einige API-Funktionen verwenden Records als Array-Elemente, die ihrerseits z. B. einen PChar enthalten. Der zeigt aber in einem solchen Fall auf eine Addresse, die ebenfalls in dem von Dir bereitgestellten Speicherblock liegt; der Block ist in einem solchen Fall also nicht einfach nur ein array von records, sondern nach dem array folgt noch ein Mischmasch von anderen Daten, die aus den Records referenziert werden. Deshalb muss man diese API-Funktionen auch in der Regel zweimal aufrufen, zuerst mal, um gesagt zu bekommen, wieviel Speicher man bereitstellen muss, dann nochmal, um den Speicherblock mit Daten gefüllt zu bekommen.

Dennis07 7. Mai 2019 17:24

AW: Buffer bei API Strukturlisten - reicht Freemem(Buffer) aus?
 
Vielleicht solltest du dir erstmal darüber klar werden, was genau du jetzt Zuweisen/Freigeben willst. Denn Enum, Record, List und Array sind ja nicht das selbe.

Zitat:

Zitat von BitSum (Beitrag 1431731)
Man muss ja i.d.R. Speicher für ein Strukturelement per pElement := AllocMem(Buffersize) reservieren.

Nö. Wie schon beschrieben, kannst du es, solltest du aber nicht. Besser ist idF
Delphi-Quellcode:
New()
und
Delphi-Quellcode:
Dispose()
für Records (da hier der Konstruktor aufgerufen wird), sowie der Standard-Arrayconstructor bei Arrays.

Zitat:

Zitat von BitSum (Beitrag 1431731)
aber reicht es dann einfach am Ende den Speicher per FreeMem(pElement) freizugeben damit die ganze Liste freigegeben wird oder muss ich mich um die Freigabe jedes Listenelements selber kümmern?

Ich weiß jetzt nicht genau, was für "Listenelemente" du meinst, aber prinzipiell werden unterobjekte bei
Delphi-Quellcode:
FreeMem()
überhaupt nicht und bei
Delphi-Quellcode:
Dispose()
nur dann freigegeben, wenn sie "gemanaged" (heißt konkret: Referenzgezählt) sind. Zeigertypen (dyn. Arrays, Klasseninstanzen, Record-Objektzeiger sowie klassische Zeiger) sind nie gemanaged.

Allerdings hier auch nochmal für den Fall, dass du es nicht wusstest: Benutze in API-Funktionen niemals gemanagte Typen, sonden wenn dann nur Zeiger auf diese. Auch Klassen (wie Listen) solltest du nicht verwenden.

BitSum 15. Mai 2019 15:08

AW: Buffer bei API Strukturlisten - reicht Freemem(Buffer) aus?
 
Vielen Dank für die Antworten, ja ich meinte mit Listenelement Pointer auf ein Struktur z.b. EnumJobs (JOB_INFO_1) vom Windows Spooler.


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