Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi FreeMem und Pointer (https://www.delphipraxis.net/47250-freemem-und-pointer.html)

oXmoX 8. Jun 2005 11:49


FreeMem und Pointer
 
Hallo zusammen!

Hab grad ein Problem mit Pointern und komme einfach nicht weiter.
Ich greife in meiner Applikation auf eine C-Library (OpenCV) zu und in C sind (wenn ich das richtig verstehe) normale Arrays implizit typisierte Pointer, denen ein entsprechend großer Speicherbereich zugewiesen wird. In C kann man ja typisierte Pointer auch so wie Arrays ansprechen ...also mit pointerName[index] zum Beispiel. Da soetwas in Delphi nicht geht habe ich mir folgende Hilfsfunktionen gebaut, mit der in komfortablen Zugriff auf die Array- (bzw. eigentlich Pointer-) Elemente habe:

Delphi-Quellcode:
procedure SetElement64d(Vector: PDouble; Index: Integer; Value: Double);
begin
  Inc(Vector, Index);
  Vector^ := Value;
  Dec(Vector, Index);
end;

function GetElement64d(Vector: PDouble; Index: Integer): Double;
begin
  Inc(Vector, Index);
  Result := Vector^;
  Dec(Vector, Index);
end;
(Es scheint übrigns keinen Unterschied zu machen, ob ich Vector als Referenz (also mit var) ober per Call By Value übergebe!)
Natürlich funktioniert das ganze nur, wenn man zuvor die benötigte Menge Speicher allokiert, z.B.:

Delphi-Quellcode:
var
  TestVector: PDouble;
begin
  GetMem(TestVector, SizeOf(PDouble) * 4000);

  SetElement64d(TestVector, 10, 0.1);
  assert(GetElement64d(TestVector, 10) = 0.1);

  FreeMem(TestVector, SizeOf(PDouble) * 4000);
end;
Meine Frage ist jetzt:
Warum bekomme ich eine EInvalidPointer Exception bei der Freigabe des Speichers in der letzten Zeile (das assert macht keine Probleme)? Besonders seltsam ist: Das ganze hatte ich zuvor schonmal für Single-Pointer (also 32-Bit-Float) geschrieben. Da hat es funktioniert. Noch schlimmer: auch die 64-Bit-Double-Version funktioniert manchmal ...wenn ich an völlig anderer Stelle in meiner Applikation Änderungen vornehme, die eigentlich absolut nichts damit zu tun haben.

Ich sitze jetzt schon mehrere Tage an dem Problem und meine Verzweiflung könnte nicht größer sein . Mal sehen ob ich ich wenigstens einen Tip bekomme.

Ach ja, ich verwende übrigens Delphi 7.0 ...könnte es möglicherweise durch ein Update behoben werden??

Gruß,
Jan

oXmoX 8. Jun 2005 11:52

Re: FreeMem und Pointer
 
...hier nochmal eine Unit zum selber ausprobieren:

Delphi-Quellcode:
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls;

type
  TForm1 = class(TForm)
    Test: TButton;
    procedure TestClick(Sender: TObject);
  private
    procedure SetElement(Vector: PDouble; Index: Integer; Value: Double);
    function GetElement(Vector: PDouble; Index: Integer): Double;
  public
    { Public-Deklarationen }
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.SetElement(Vector: PDouble; Index: Integer; Value: Double);
begin
  Inc(Vector, Index);
  Vector^ := Value;
  Dec(Vector, Index);
end;

function TForm1.GetElement(Vector: PDouble; Index: Integer): Double;
begin
  Inc(Vector, Index);
  Result := Vector^;
  Dec(Vector, Index);
end;

procedure TForm1.TestClick(Sender: TObject);
var
  TestVector: PDouble;
begin
  GetMem(TestVector, 20);

  Self.SetElement(TestVector, 10, 0.1);
  ShowMessage(FloatToStr(Self.GetElement(TestVector, 10)));
  ShowMessage(FloatToStr(Self.GetElement(TestVector, 10)));
  assert(Self.GetElement(TestVector, 10) = 0.1);

  FreeMem(TestVector, 20);
end;

end.
Wenn ich den Code aufrufe, dann schlägt (in der obigen) Form sogar das assert schon fehl. Die erste ShowMessage gibt korrekt 0.1 aus und die zweite zeigt jedesmal 1,9059698307456E-307.
...Ich feuer gleich mein Notebook gegen die Wand

barf00s 8. Jun 2005 11:59

Re: FreeMem und Pointer
 
SizeOf(PDouble) ergibt übrigens 4, und nicht 8 wie du vllt wolltest (SizeOf(Double) = 8) ...

Basilikum 8. Jun 2005 12:01

Re: FreeMem und Pointer
 
hat jetzt nicht direkt etwas mit dem Problem zu tun, aaaaber..... ;-)

weshalb verwendest Du nicht diese meines Erachtens bequemere Variante für den Zugriff ?
Delphi-Quellcode:
Type
  PDoubles = ^TDoubles;
  TDoubles = Array[0..0] Of Double;

var
  TestVector: PDouble;
begin
  GetMem(TestVector, SizeOf(PDouble) * 4000);

  PDoubles(TestVector)^[10]:=0.1;
  assert(PDoubles(TestVector)^[10] = 0.1);

  FreeMem(TestVector, SizeOf(PDouble) * 4000);
end;
oder aber TestVector direkt als PDoubles deklarierer - hat nichts mit Delphi's Dyn-Array zu tun

sofern Delphi's Double generell Binary-Kompatibel zu C ist, funktioniert dies einwandfrei

Delphi-Quellcode:
procedure TForm1.TestClick(Sender: TObject);
var
  TestVector: PDouble;
begin
  GetMem(TestVector, 20);

  Self.SetElement(TestVector, 10, 0.1);
  ShowMessage(FloatToStr(Self.GetElement(TestVector, 10)));
  ShowMessage(FloatToStr(Self.GetElement(TestVector, 10)));
  assert(Self.GetElement(TestVector, 10) = 0.1);

  FreeMem(TestVector, 20);
end;
ich weiss jetzt nicht im Detail, wie gross ein Double im Speicher ist, aber 20 Bytes reichen für 11 Doubles sicherlich nicht...... (GetMem(TestVector, 20 * SizeOf(Double));

barf00s 8. Jun 2005 12:02

Re: FreeMem und Pointer
 
schonwieder SizeOf(PIrgendwasWasAufNenPointerZeigt)

oXmoX 8. Jun 2005 12:06

Re: FreeMem und Pointer
 
Zitat:

Zitat von barf00s
SizeOf(PDouble) ergibt übrigens 4, und nicht 8 wie du vllt wolltest (SizeOf(Double) = 8) ...

Oops ..da hat sich wohl ein kleiner Fehler eingeschlichen. Dennoch behebt das leider nicht mein Problem.

Trotzdem danke für den Tip.
Hier die korrigierte Version der Unit.

Delphi-Quellcode:
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls;

type
  TForm1 = class(TForm)
    Test: TButton;
    procedure TestClick(Sender: TObject);
  private
    procedure SetElement(Vector: PDouble; Index: Integer; Value: Double);
    function GetElement(Vector: PDouble; Index: Integer): Double;
  public
    { Public-Deklarationen } 
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm} 

procedure TForm1.SetElement(Vector: PDouble; Index: Integer; Value: Double);
begin
  Inc(Vector, Index);
  Vector^ := Value;
  Dec(Vector, Index);
end;

function TForm1.GetElement(Vector: PDouble; Index: Integer): Double;
begin
  Inc(Vector, Index);
  Result := Vector^;
  Dec(Vector, Index);
end;

procedure TForm1.TestClick(Sender: TObject);
var
  TestVector: PDouble;
begin
  GetMem(TestVector, 20 * SizeOf(PDouble));

  Self.SetElement(TestVector, 10, 0.1);
  ShowMessage(FloatToStr(Self.GetElement(TestVector, 10)));
  ShowMessage(FloatToStr(Self.GetElement(TestVector, 10)));
  assert(Self.GetElement(TestVector, 10) = 0.1);

  FreeMem(TestVector, 20 * SizeOf(PDouble));
end;

end.
Hat sonst noch jemand ne Idee. Ich hab auch schon versucht die Code-Optimierung im Compiler auszuschalten.

Basilikum 8. Jun 2005 12:07

Re: FreeMem und Pointer
 
Delphi-Quellcode:
GetMem(TestVector, 20 * SizeOf(PDouble));
korrekt wäre:
Delphi-Quellcode:
GetMem(TestVector, 20 * SizeOf(Double));

oXmoX 8. Jun 2005 12:11

Re: FreeMem und Pointer
 
Zitat:

Zitat von Basilikum
Delphi-Quellcode:
GetMem(TestVector, 20 * SizeOf(PDouble));
korrekt wäre:
Delphi-Quellcode:
GetMem(TestVector, 20 * SizeOf(Double));

OK. Vielen Dank. Dann bin ich jetzt wieder etwas schlauer geworden. Dennoch löst das mein Problem immernoch nicht. Die Zusicherung schlägt auch jetzt noch fehl.

oXmoX 8. Jun 2005 12:22

Re: FreeMem und Pointer
 
Sorry, dass ich das Ding jetzt schon wieder korrigieren muss ...nur der Vollständigkeit halber nochmal mit (hofentlich) korrekter Speicher-Allokierung:

Delphi-Quellcode:
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls;

type
  TForm1 = class(TForm)
    Test: TButton;
    procedure TestClick(Sender: TObject);
  private
    procedure SetElement(Vector: PDouble; Index: Integer; Value: Double);
    function GetElement(Vector: PDouble; Index: Integer): Double;
  public
    { Public-Deklarationen } 
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm} 

procedure TForm1.SetElement(Vector: PDouble; Index: Integer; Value: Double);
begin
  Inc(Vector, Index);
  Vector^ := Value;
  Dec(Vector, Index);
end;

function TForm1.GetElement(Vector: PDouble; Index: Integer): Double;
begin
  Inc(Vector, Index);
  Result := Vector^;
  Dec(Vector, Index);
end;

procedure TForm1.TestClick(Sender: TObject);
var
  TestVector: PDouble;
begin
  GetMem(TestVector, 20 * SizeOf(Double));

  Self.SetElement(TestVector, 10, 0.1);
  assert(Self.GetElement(TestVector, 10) = 0.1);

  FreeMem(TestVector, 20 * SizeOf(Double));
end;

end.
...und wie gesagt: das Assert schlägt fehl!

himitsu 8. Jun 2005 12:34

Re: FreeMem und Pointer
 
Das/Warum SizeOf(PDouble) ja 4 ist, ist ja geklärt.

Ich wollte nur noch sagen, dass man es auch anders hätte lösen können ^^

Delphi-Quellcode:
SizeOf(TestVector^)
das ^ macht ja aus dem Zeige wieder den Datentyp, was
Delphi-Quellcode:
SizeOf(PDouble^)
und damit natürlich auch
Delphi-Quellcode:
SizeOf(Double)
entspricht, allerdings hätte dieses auch mal den Vorteil, wenn der Typ von TestVector geändert würde, denn dann würde das Ergebnis von SizeOf immernoch stimmen ^^

marabu 8. Jun 2005 12:42

Re: FreeMem und Pointer
 
Hallo Jan,

du solltest Non-Integer-Values nicht auf Gleichheit testen, ein epsilon-Test ist da sicherer:
Delphi-Quellcode:
var
  epsilon: Double = 0.001;

procedure TForm1.TestClick(Sender: TObject);
var
  TestVector: PDouble;
  allocSize: integer;
begin
  allocSize := 20 * SizeOf(Double);
  GetMem(TestVector, allocSize);

  SetElement(TestVector, 10, 0.1);
  assert(Abs(GetElement(TestVector, 10) - 0.1) < epsilon);

  FreeMem(TestVector, allocSize);
end;
Und die folgende Zeile kannst du dir bei Call-By-Value schenken:

Delphi-Quellcode:
Dec(Vector, Index);
Grüße vom marabu

oXmoX 8. Jun 2005 12:42

Re: FreeMem und Pointer
 
Zitat:

Zitat von himitsu
Das/Warum SizeOf(PDouble) ja 4 ist, ist ja geklärt.

Ich wollte nur noch sagen, dass man es auch anders hätte lösen können ^^

Delphi-Quellcode:
SizeOf(TestVector^)
das ^ macht ja aus dem Zeige wieder den Datentyp, was
Delphi-Quellcode:
SizeOf(PDouble^)
und damit natürlich auch
Delphi-Quellcode:
SizeOf(Double)
entspricht, allerdings hätte dieses auch mal den Vorteil, wenn der Typ von TestVector geändert würde, denn dann würde das Ergebnis von SizeOf immernoch stimmen ^^

..ist mir schon klar. Trotzdem vielen Dank!

Hab grad neu Lösung gefunden (bzw. gefunden bekommen in nem anderen Forum). Wenn man statt es expliziten Wertes 0.1 eine Variable (Type Double) nimmt geht's nämlich auf einmal. 0.1 ist nämlich implizit Single.


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