Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi Zeigerproblem (https://www.delphipraxis.net/111529-zeigerproblem.html)

nEmai 4. Apr 2008 20:25


Zeigerproblem
 
Hiho,

Ich hab es (noch) nicht so mit Zeigern und daher keine Ahnung was Delphi nun wirklich von mir will.

Delphi-Quellcode:
  TBarGraph = class
    private
      PBitmap: ^TBitmap;
      FBars: Array of Integer;
      procedure Bar(const aPosition, aValue: Integer);
      procedure ArraySwap(const aPosition1, aPosition2: Integer);
      procedure BitmapSwap(const aPosition1, aPosition2: Integer);
    public
      constructor Create(var aImage: TImage; aForm: TForm);
      destructor Destroy; override;
      procedure Draw;
      procedure BubbleSort;
      procedure QuickSort(const aLow, aHigh: Integer);
  end;

var
  Form1: TForm1;
  bKill: Boolean;
  MyBarGraph: TBarGraph;

implementation
{$R *.dfm}

constructor TBarGraph.Create(var aImage: TImage; aForm: TForm);
const
  MAXRANDOM= 530;
  BARQUANTITY= 150;
var
  iCounter: Integer;
begin
  inherited Create;
  setLength(FBars, BARQUANTITY);
  for iCounter:= 0 to High(FBars) do
    FBars[iCounter]:= random(MAXRANDOM)+1;
  PBitmap:= @aImage.Picture.Bitmap; //<--------
  PBitmap^.Height:= MAXRANDOM+20;
  PBitmap^.Width:= (Length(FBars)*4)+8;
  aImage.Height:= PBitmap^.Height;
  aImage.Width:= PBitmap^.Width;
  aForm.Height:= PBitmap^.Height+130;
  aForm.Width:= PBitmap^.Width+100;
end;
Schmeißt bei Aufruf in der markierten Zeile den Fehler "Variable erforderlich".
Was ist daran falsch?

MfG.

Muetze1 4. Apr 2008 20:26

Re: Zeigerproblem
 
Wozu brauchst du den Zeiger? Schmeiss das ^ bei der Deklaration raus und das @ bei der Zuweisung. Instanzenvariablen sind intern schon Zeiger, somit alles wie gehabt und keine Verrenkungen nötig.

nEmai 4. Apr 2008 20:51

Re: Zeigerproblem
 
Und wenn ich es doch so machen wollte?
Ich will ja was bei lernen. Und dabei mein ich nicht unbedingt wie man es am besten macht. ^^

Muetze1 4. Apr 2008 21:01

Re: Zeigerproblem
 
Zitat:

Zitat von nEmai
Und wenn ich es doch so machen wollte?
Ich will ja was bei lernen. Und dabei mein ich nicht unbedingt wie man es am besten macht. ^^

Ok, aber dann bleibt die Frage: Warum ein Zeiger? Du versuchst einen Zeiger auf einen Zeiger zu bauen und dazu fällt mir gar keine Nutzung für ein.

Und zu deinem Problem Bitmap ist eine Property und somit kannst du davon keine Adresse ermitteln. Aus dem einfachen Grund da eine Property mit einem Getter, Setter oder einer direkten Variablenreferenz definiert werden kann. Somit kann er keine Adresse ermitteln.

taaktaak 4. Apr 2008 21:05

Re: Zeigerproblem
 
Wenn du etwas über die Verwendung von Zeigern lernen möchtest, dann suche dir besser einen anderen Anwendungsfall: Vielleicht eine einfach oder doppelt verkettete Liste, da macht es Sinn und ist auch didaktisch besser (denke ich mir mal so)

nEmai 4. Apr 2008 21:17

Re: Zeigerproblem
 
Nun ich wollte hier Zeiger einsetzen, weil ich mir die Bitmap an Image.Picture als Objekt TBitmap vorgestellt hab, mit dem ich arbeiten will.
Meine Logik sagt mir ich brauch nen Pointer darauf.
Aber ich hab es nun so gemacht und es geht. ^^

Delphi-Quellcode:
PBitmap: TBitmap;
..
PBitmap:= aImage.Picture.Bitmap;
Danke&MfG^^

Muetze1 4. Apr 2008 21:20

Re: Zeigerproblem
 
Zitat:

Zitat von nEmai
Meine Logik sagt mir ich brauch nen Pointer darauf.

Deine Logik ist richtig. Füge deinem Wissen einfach noch folgendes hinzu:

Zitat:

Zitat von Muetze1
Instanzenvariablen sind intern schon Zeiger, ...

Damit hast du deinen Pointer und deine Logik ist vollkommen richtig.

nEmai 4. Apr 2008 21:54

Re: Zeigerproblem
 
Ok, das heißt, wenn ich es richtig versteh:
Image.Picture.Bitmap ist ein Pointer auf eine Bitmap, also eine Speicheradresse.

Dann müsste
Delphi-Quellcode:
var
  ptrBitmap: ^TBitmap;
begin
  ptrBitmap:= Image.Picture.Bitmap;
end;
ja funktionieren.
Es kommt aber inkompatible Typen Pointer und TBitmap.
Wird das schon intern dereferenziert und wenn ja gibt es einen Befehl mit dem man es verhindern kann?
(Ich wüsste einfach gern ob und wie ich darauf einen eigenen Pointer machen kann.)

MfG^^

Muetze1 4. Apr 2008 22:29

Re: Zeigerproblem
 
Zitat:

Zitat von nEmai
Ok, das heißt, wenn ich es richtig versteh:
Image.Picture.Bitmap ist ein Pointer auf eine Bitmap, also eine Speicheradresse.

Dann müsste
Delphi-Quellcode:
var
  ptrBitmap: ^TBitmap;
begin
  ptrBitmap:= Image.Picture.Bitmap;
end;
ja funktionieren.
Es kommt aber inkompatible Typen Pointer und TBitmap.
Wird das schon intern dereferenziert und wenn ja gibt es einen Befehl mit dem man es verhindern kann?
(Ich wüsste einfach gern ob und wie ich darauf einen eigenen Pointer machen kann.)

MfG^^

Jain. Mal ein wenig ausführlicher:

Wir nehmen uns ein Objekt:

Delphi-Quellcode:
var
  a: TObject;
Dann hast du dort an der Stelle von a nicht direkt das komplette TObject mit allen seinen Methoden und Feldern sondern nur einen Zeiger. Dieser würde dann auf den Speicherbereich zeigen, wo die Felder, etc liegen. Wenn du diese Variable deklariert hast, kannst du ja mal mit SizeOf(a) deren Grösse ermitteln: du wirst 4 bekommen. 4 Bytes, also genau die Grösse eines Pointers.

Du musst, um das Objekt nutzen zu können, die eine Instanz erstellen. Diese erhälst du mit dem Konstruktor-Aufruf. Dieser reserviert den notwendigen Speicher für das Objekt und gibt dir den Zeiger genau auf diesen Speicherbereich zurück. Von daher kannst du es auch abfragen ob ein Objekt existiert, wenn die Variable ordentlich intialisiert wird. Du kannst mit einem Vergleich auf <> nil abprüfen, ob schon eine Adresse drinne steht (wie bei einem Pointer btw).

Deshalb haben auch viele Probleme mit den Objekten, wenn sie Free aufrufen und danach weiterhin auf das Objekt zugreifen und AV's bekommen und selbst Assigned() bzw. ein Vergleich auf Nil nichts bringt: Der Zeiger zeigt weiterhin auf die Adresse (hat ihm ja keiner was anderes gesagt), aber da ist nichts mehr. Somit: FreeAndNil() ruft Free auf und setzt den Zeiger auf NIL und somit klappen die Abfragen auch.

Anderes Beispiel:

Delphi-Quellcode:
var
  a, b: TStrings;
begin
  a := TStringList.Create; // neue Instanz
  b := a;                  // Objekt wird nicht kopiert, nur der Zeiger!

  a.Text := 'erstes Object';

  ShowMessage(b.Text);
end;
Bei dem Beispiel erzeugst du nur ein Objekt. Wäre a und b jeweils intern kein Zeiger, dann müsste er das komplette Objekt kopieren, da es komplett eigene Speicherbereiche sind. Wenn dem so ist, dann müsste die ShowMessage() einen leeren String ausgeben. Tut er aber nicht, sondern er gibt "erstes Object" aus. Grund dafür: er kopiert nur die Adresse aus dem Pointer in den anderen. Somit zeigen beide auf das selbe Objekt im Speicher.

Also: Wenn du eine Objektvariable deklarierst, ist diese nur 4 Bytes gross, da es ein Zeiger ist.

Dies ist das interne Handling, somit geschieht auch die Dereferenzierung automatisch intern von Delphi. Und ausserdem ist Delphi recht typensicher (wie oft habe ich das in den letzten Tagen geschrieben...) und TBitmap und Pointer sind nunmal unterschiedliche Typen.

Namenloser 4. Apr 2008 22:33

Re: Zeigerproblem
 
Nimm zum Üben einen Record bzw einen Pointer auf diesen Record. Objekte sind zum Üben von Pointern denkbar ungeeignet.

gmc616 5. Apr 2008 03:31

Re: Zeigerproblem
 
Für mich, der grad aus der Kneipe kommt und aus lauter Frust ein .. zwei Bier zu viel getrunken hat, weil er nicht ein einzigstes Dartspiel gewonnen hat (man, und ich war mal so gut darin), stellt sich die Frage:
Willst du was über Zeiger lernen, oder des Problem lösen.

Für die Lösung hätte ich einen Tipp.
Lass das ganze Zeiger-gemurks (^ und @) weg. Dein Parameter aImage ist sowieso ein Zeiger auf das Original-Image, so wie Muetze1 schreibt.
Mit dem kannst du genauso weiter arbeiten, als wenn du das Image "direkt in der Form" ansprichst. Das ist nämlich letzten Endes eben auch nur ein Zeiger.

Für das, was ich deinem Code entnehmen kann reicht das völlig.


Zitat:

Zitat von nEmai
Ok, das heißt, wenn ich es richtig versteh:
Image.Picture.Bitmap ist ein Pointer auf eine Bitmap, also eine Speicheradresse.

Dann müsste
Delphi-Quellcode:
var
  ptrBitmap: ^TBitmap;
begin
  ptrBitmap:= Image.Picture.Bitmap;
end;
ja funktionieren.

Ähm ... :gruebel: ... Nein. Das wäre ja quasi ein Zeiger auf einen Zeiger.

Delphi-Quellcode:
var
  oBitmap: TBitmap;
begin
  oBitmap:= Image.Picture.Bitmap;
end;
... würde reichen. Du kopierst nur den Zeiger. Das ist doch das was du vor hast, oder?

Bitte nicht übel nehmen, aber ich glaube, du hast den Begriff "Instanz" nicht richtig vereinnahmt.

TImage ist die Klasse.

mit
Delphi-Quellcode:
aImage := TImage.create (??);
instanzierst du die Klasse.
aImage ist also eine Instanz der Klasse TImage, was eigentlich nur ein Zeiger ist (siehe Muetze1).

Alle Unterobjekte von aImage sind auch nur Instanzen (Zeiger) der entsprechenden Klassen.

So sollte es eigentlich funktionieren:
Delphi-Quellcode:
constructor TBarGraph.Create(var aImage: TImage; aForm: TForm);
const
  MAXRANDOM= 530;
  BARQUANTITY= 150;
var
  iCounter: Integer;
begin
  inherited Create;
  setLength(FBars, BARQUANTITY);

  for iCounter:= 0 to High(FBars) do
    FBars[iCounter]:= random(MAXRANDOM)+1;
 
  aImage.Picture.Bitmap.Height:= MAXRANDOM+20;
  aImage.Picture.Bitmap.Width:= (Length(FBars)*4)+8;
 
  aImage.Height:= PaImage.Picture.Bitmap.Height;
  aImage.Width:= aImage.Picture.Bitmap.Width;
 
  aForm.Height:= aImage.Picture.Bitmap.Height+130;
  aForm.Width:= aImage.Picture.Bitmap.Width+100;
end;
Wenn du Objekte (also alles das, was von TObject erbt) in eine Procedure/Funktion übergibst, übergibst du eigentlich nur den Zeiger auf das Objekt.

das ist das gleiche wie zB.
Delphi-Quellcode:
 procedure HoleBier (Var aGlas : Integer); // Zeiger auf Variable wird kopiert
begin
  if aGlas = 0 then
   aGlas := 100;
end;
Ich gebe dir mein leeres Bierglas mit, und du kommst mit dem _selbem_ Glas wieder zurück, nur das es jetzt voll ist.

Mit folgender Funktion würdest du mit _meinem_ Glas wieder zurück kommen, es wäre aber weiterhin leer.
Delphi-Quellcode:
 procedure HoleBier (aGlas : Integer); // Variable wird kopiert
begin
  if aGlas = 0 then
   aGlas := 100;
end;
Was mit dem Bier passiert ist weiß ich nicht. Es ist halt weg.

Da aber Glas eigentlich ein TObject ist (sein sollte), welches unter anderem das Property Inhalt hat, kommst du mit folgender Funktion:
Delphi-Quellcode:
 
type TGlas = Class (TObject);
  public
    Property Inhalt : Integer;
[...]

procedure HoleBier (aGlas : TGlas); // Zeiger auf Object wird kopiert
begin
  if aGlas.Inhalt = 0 then
   aGlas.Inhalt := 100;
end;
mit _meinem_ Glas wieder, welches aber wieder voll wäre.

Mit TImage ist es genau das gleiche.

Ich hoffe, ich hab keinen Quatsch erzählt ... :cheers:

Larsi 5. Apr 2008 03:40

Re: Zeigerproblem
 
[OT] Warum seid Ihr alle noch online? :) [OT]

nEmai 5. Apr 2008 13:14

Re: Zeigerproblem
 
So, nachdem ich mich durch die vielen Informationen durchgearbeitet hab..

@Muetze1:
FreeAndNil() kennt mein Delphi 2005 nicht.

@gmc616:
Eigentlich dachte ich schon, dass mir klar ist, was eine Instanz ist.
Dein Konstruktorvorschlag umgeht das Thema des Threads. oO
(Ich brauch die Variable in der Klasse.)

@Beide:
Das einzige Problem, das ich hatte, war offenbar der Umgang mit Objektvariablen.
Mein alter Gedankengang etwas genauer:
Delphi-Quellcode:
var
  ptrBitmap: ^TBitmap;
begin
  Image.Picture.Bitmap = 0x12345 //Bsp.Adresse
  ptrBitmap = null

  ptrBitmap:= Image.Picture.Bitmap; //null <-überschreibt- 0x12345

  Image.Picture.Bitmap = 0x12345
  ptrBitmap = 0x12345
end;
Aber dank den guten Erklärungen hab ich es jetzt denk ich richtig verstanden. ^^

MfG.

Muetze1 5. Apr 2008 13:29

Re: Zeigerproblem
 
Zitat:

Zitat von nEmai
@Muetze1:
FreeAndNil() kennt mein Delphi 2005 nicht.

Also Delphi 4, Delphi 5, Delphi 6, Delphi 7 sowie BDS2006 kennen diese Funktion in der SysUtils. Mit diesen Delphi Versionen habe ich schon gearbeitet und weiß das die Funktion definitiv vorhanden ist. Schon allein da BDS2006 nach Delphi 2005 herauskam, ist es für mich schwer zu glauben, dass es diese Funktion in der Version nicht mehr geben sollte.

mkinzler 5. Apr 2008 13:46

Re: Zeigerproblem
 
@nEmai: Nochmal TBitmap ist schonn ein Zeiger (Referenz)!!!!!!!!

nEmai 5. Apr 2008 14:10

Re: Zeigerproblem
 
@Muetze1
Delphi-Quellcode:
var
  Bitmap: TBitmap;
begin
  Bitmap:= TBitmap.Create;
 
  Bitmap.FreeImage;
  Bitmap.Free;
  Bitmap.FreeInstance;
end;
Das ist alles was mir an dem Bsp. mit free angezeigt wird, bei FreeAndNil Compilierfehler.

Zitat:

Zitat von mkinzler
@nEmai: Nochmal TBitmap ist schonn ein Zeiger (Referenz)!!!!!!!!

Zitat:

Zitat von nEmai
Aber dank den guten Erklärungen hab ich es jetzt denk ich richtig verstanden. ^^

Is ja recht... oO

mkinzler 5. Apr 2008 14:36

Re: Zeigerproblem
 
FreeAnNil() ist eine Prozedur und keine Methode:

Delphi-Quellcode:
FreeAndNil( Bitmap);


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