AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Data-Pointer in Interfacevariable casten

Ein Thema von RedOne · begonnen am 14. Jul 2010 · letzter Beitrag vom 16. Jul 2010
Antwort Antwort
RedOne

Registriert seit: 2. Jun 2008
71 Beiträge
 
Delphi XE2 Professional
 
#1

Data-Pointer in Interfacevariable casten

  Alt 14. Jul 2010, 10:19
Hallo zusammen
Die Delphi-Interfaces bringen mich gerade an den Rand der verzweiflung...

Ich habe folgenden Code:

Delphi-Quellcode:
var
  i: IDataObject;
begin
  if ListViewMain.ItemIndex >= 0 then begin
    i:= ListViewMain.Selected.Data as IDataObject;
    FillForm( i );
  end
end;
Im Data werden verschiedene Objekte abgelegt die alle dieses Interface implementieren.
Dieser Code gibt ein Compillerfehler (E2015 Operator ist auf diesen Operandentyp nicht anwendbar).
Caste ich mit IDataObject( ListViewMain.Selected.Data ) gibt es mir einen Laufzeitfehler (Data ist Assigned).

Wo liegt den hier das Problem?
Merci
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.326 Beiträge
 
Delphi 12 Athens
 
#2

AW: Data-Pointer in Interfacevariable casten

  Alt 14. Jul 2010, 11:15
Ein Pointer ist halt kein Interface.

Delphi-Quellcode:
i := IDataObject(ListViewMain.Selected.Data);
// oder
i := Interface(ListViewMain.Selected.Data) as IDataObject;

PS: Wie regelst du denn die Referenzzählung?
Wenn keine andereren Referenzen auf das Interface bestehen und du dieses nicht beachtest, dann würde das Interface freigegeben, auch wenn du dieses hier verlinkt hast, falls du keine Referenz hierfür beantragst.
Delphi-Quellcode:
// zuweisen
ListViewMain.Selected.Data := Pointer(DataObject);
DataObject._AddRef;

// vorm freigeben
DataObject.__Release;
Ein Therapeut entspricht 1024 Gigapeut.
  Mit Zitat antworten Zitat
RedOne

Registriert seit: 2. Jun 2008
71 Beiträge
 
Delphi XE2 Professional
 
#3

AW: Data-Pointer in Interfacevariable casten

  Alt 14. Jul 2010, 14:08
Merci für Deine Antwort.
Mit i:= IDataObject( ListViewMain.Selected.Data ); wird bei dieser Zeile eine Exception geworfen (Zugriffsverletzung) obwohl das Objekt existiert und auch instanziert ist (habs zur Sicherheit nochmals getestet).
Casten mit i:= Interface(ListViewMain.Selected.Data) as IDataObject; ergibt einen Compilerfehler...

Das Objekt (TAccountDataExtend) im Data implementiert eine weitere Klasse (TAccountData) und das Interface. TInterfacedObject implementiere ich schon auf der implementierten Klasse TAccountData. Ich nehme an, dies sollte schon funktionieren? Ein anderer Weg ist mir (bisher) noch nicht bekannt.

Delphi-Quellcode:
TAccountData = class( TInterfacedObject )[...]
TAccountDataExtend = class( TAccountData, IDataObject ) [...]

Die Frage betreffend der Referenzzählung ist gut
In Delphi habe ich bisher nicht das Glück gehabt mit Interfaces zu arbeiten. Da bin ich wohl ein bisschen .Net verwöhnt bei dem dies keine Rolle spielt

Herzlichen Dank
RedOne
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.326 Beiträge
 
Delphi 12 Athens
 
#4

AW: Data-Pointer in Interfacevariable casten

  Alt 14. Jul 2010, 14:22
Ups, meinte natürlich IInterface
i:= IInterface(ListViewMain.Selected.Data) as IDataObject; .

Es sah allerdings so aus, als wenn das Interface direkt im .Data drin ist, bzw. es klang irgendwie danach.
Aber wenn es das Objekt ist, dann mußt du .Data natürlich in Diese casten.

i := TAccountDataExtend(ListViewMain.Selected.Data) as IDataObject;
Ein Therapeut entspricht 1024 Gigapeut.

Geändert von himitsu (14. Jul 2010 um 14:24 Uhr)
  Mit Zitat antworten Zitat
RedOne

Registriert seit: 2. Jun 2008
71 Beiträge
 
Delphi XE2 Professional
 
#5

AW: Data-Pointer in Interfacevariable casten

  Alt 14. Jul 2010, 14:35
Das geht ja flux mit Antworten... Merci!

i:= IInterface(ListViewMain.Selected.Data) as IDataObject; compilliert zwar aber gibt immer noch einen Laufzeitfehler.
Gerne würde ich direkt in mein TAccountData casten, kann ich aber nicht:
Ich habe ein abstraktes Basisformular da alle Editierfenster den gleichen Aufbau besitzen.
Nun habe ich ein AccountForm davon abgeleitet. Dieses rufe ich auf und fülle beim Start eine Listview (auf Basisklasse vorhanden) mit meinen TAccountDataExtend's.
Wird ein Datensatz angeklickt und auf den Löschbutton geklickt, so wird auf Stufe des Basisformulares, dass keine Ahnung hat um was für ein Objekt es sich handelt, das Objekt im Data auf mein IDataObject-Interface gecastet und die Funktion "Delete" aufgerufen, die wegen dieses Interfaces auf meinem TAccountDataExtend implementiert sein muss.
Aber auf Stufe des Basisformulares weiss ich nicht was für ein konkretes Objekt sich in der Listview befindet. Allerdings weiss ich dass jedes Objekt die genannte "Delete"-Funktion implementiert, da diese ja von diesem Interface abgeleitet sind.

Wenn ich die Klasse für TAccountDataExtend zu Testzwecken auf Stufe des Basisformulars einbinde und caste sehe ich dass dieses Objekt existiert und kann auch die Deletefunktion aufrufen. Darum bin ich ein bisschen verwirrt wieso ich mein Data-Pointer nicht in dieses Interface casten kann, respektive wieso mir dieser Hund eine Exception wirft

Herzlichen Dank

Ps.
Einige Auszüge aus dem Code. Das beim "OnChange"-Ereignis gelöscht wird macht weniger Sinn. Dies ist mehr zum Verständnis
Delphi-Quellcode:
//-------- TFormEditBase
TFormEditBase = class( TForm )
  procedure ListViewMainChange( _sender: TObject; _item: TListItem; _change: TItemChange );
  ...
end;

procedure TFormEditBase.ListViewMainChange( _sender: TObject; _item: TListItem; _change: TItemChange );
var
  i: IDataObject;
begin
  if ListViewMain.ItemIndex >= 0 then begin
    i:= IInterface(ListViewMain.Selected.Data) as IDataObject;
    i.Delete;
  end;
end {ListViewMainChange};

//-------- TFormAccounts
TFormAccounts = class( TFormEditBase )
  procedure Load;
  ...
end;

procedure TFormAccounts.Load;
var
  tmp: TAccountDataExtend;
  obj: TObjectList;
  i: Integer;
  item: TListItem;
begin
  GetAllAccounts( obj );
  try
    for i:= 0 to obj.Count - 1 do begin
      tmp:= obj[ i ] as TAccountDataExtend;
      item:= ListViewMain.Items.Add;
      item.SubItems.Add( tmp.Name );
      item.Data:= tmp;
    end;
  finally
    obj.Free;
  end;
end;

Geändert von RedOne (14. Jul 2010 um 14:42 Uhr)
  Mit Zitat antworten Zitat
brechi

Registriert seit: 30. Jan 2004
823 Beiträge
 
#6

AW: Data-Pointer in Interfacevariable casten

  Alt 14. Jul 2010, 19:40
Erstmal grundsätzlich: Speicher sollte dort freigegeben werden wo man diesen auch initialisiert hat

Delphi-Quellcode:
var
  obj: TObjectList;
begin
  obj := TObjectList.Create;
  try
    GetAllAccounts( obj );
    ...
  finally
    obj.Free;
  end;
end;
Dann zu deinem Problem:

1) warum verwendest du eine Objektlist wenn du mit Interfaces arbeitest? Gibt auch eine TInterfaceList
2) wenn du eine ObjectList verwendest, ist auch OwnsObjects auf false, sonst sind die Objekte beim Free alle wieder weg (auch deine TInterfaceObjects)

Bei Item.Data solltest du dann auch das Interface angeben und den Referenzcounter erhöhen. Hier mal ein Beispiel hoffe das ist jetzt so korrekt und du kannst was damit anfangen

Delphi-Quellcode:

type
  IIrgendwas = interface ['{0EF386E9-7F06-4D9A-A261-37D61BE3F8BE}']
    procedure MessageBox;
  end;

  TMyObject = class(TInterfacedObject, IIrgendwas)
    procedure MessageBox;
  end;

procedure TForm6.FormCreate(Sender: TObject);
var
  x: TInterfaceList;
  obj, obj2: IInterface;
  obj3: IIrgendwas;
  sl: TStringList;
  i: integer;
begin
  obj := TMyObject.Create;

  sl := TStringList.Create;
  try
    x := TInterfaceList.Create; // statt der ObjectList
    try
      x.Add(obj);
      for i := 0 to x.Count - 1 do begin
        sl.Add('test');
        obj._AddRef; // ein cast als Pointer bekommt Delphi nicht mit (dein data)
        sl.Objects[i] := Pointer(obj);
      end;
    finally
      FreeAndNil(x);
    end;

  obj := nil;

  // verwenden und freigeben
  obj2 := IInterface(Pointer(sl.Objects[0]));
  if Succeeded(obj2.QueryInterface(IIrgendwas, obj3)) then
    obj3.MessageBox;
  obj2._Release;
  finally
    FreeAndNil(sL);
  end;

end;

{ TMyObject }

procedure TMyObject.MessageBox;
begin
  MessageBoxA(0, 'test', nil, 0)
end;
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 06:55 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 by Thomas Breitkreuz