AGB  ·  Datenschutz  ·  Impressum  







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

Zu unbekanntem Klassenderivat casten?

Ein Thema von blackdrake · begonnen am 12. Mär 2009 · letzter Beitrag vom 17. Mär 2009
Antwort Antwort
Seite 1 von 2  1 2      
blackdrake

Registriert seit: 22. Aug 2003
Ort: Bammental
618 Beiträge
 
Delphi 10.3 Rio
 
#1

Zu unbekanntem Klassenderivat casten?

  Alt 12. Mär 2009, 23:21
Hallo.

Ich habe folgendes Problem: Ich möchte ein Objekt kopieren, weiß aber zur Laufzeit nicht, um welche Klasse es sich handelt.

Der Code sieht in etwa so aus:

Delphi-Quellcode:
type
  TBasis = class(TObject);
  TAbleitungA = class(TBasis);
  TAbleitungB = class(TBasis);

var
  test, kopie: TBasis;

begin
  test := GibtMirWas(); // gibt entweder TAbleitungA oder TAbleitungB

  // Versuche: Kopie anlegen
  {Versuch 1} kopie := test.ClassType.Create; // Problem: Wie TObject nach "test.ClassType.ClassName" casten? Ich will am Ende aber ein TAbleitungX und kein TBasis haben...
  {Versuch 1} kopie := test.NewInstance; // Problem: Das selbe

  kopie.feld := 'ABC'; // wegen dieser Veränderung darf die originale Klasse nicht verwendet werden.

  MeineFunktion(kopie);
end;
Wie kann ich das machen?

Gruß
blackdrake
Daniel Marschall
  Mit Zitat antworten Zitat
Benutzerbild von _frank_
_frank_

Registriert seit: 21. Feb 2003
Ort: Plauen / Bamberg
922 Beiträge
 
Delphi 3 Professional
 
#2

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 00:08
bei Kompoenenten geht das so wie in diesem Auszug aus meinem DFM-Editor:

Delphi-Quellcode:
function CreateComponent(aName,aType:string;Parent:TWinControl;Owner:TComponent):TComponent;
var
  classe:TpersistentClass;
  CC: TComponentClass;
begin
  result:=nil;
  classe:=getclass(AType);
  if assigned(classe) then
  begin
    CC:=TComponentClass(classe);
    result:=cc.Create(owner);
  end;
end;
bei einem reinen TObject, ist mir keine ähnliche Variante bekannt

HTH Frank
  Mit Zitat antworten Zitat
Dax
(Gast)

n/a Beiträge
 
#3

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 00:21
Du könntest deiner TBasis eine abstrakte Clone-Methode hinzufügen, die eine TBasis-Instanz zurückgibt, die eine Kopie der aufgerufenen Instanz zu sein hat. Ganz offensichtlich ist ja alles, was du ändern willst, in TBasis
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 00:22
test.ClassType.Create

das Problem hierbei ist, daß ClassType vom Typ TClass ist, was der Klassentyp von TObject ist.
leider wird hier wohl vom Compiler .Create mit dem "Variablen"-Type hat verlinkt, also TClass.Create verwendet und demnach wird ein TObject erstellt.



meine schnelle Lösung war es eine "neue" Function einzuführen siehe Function CreateInstance in
http://www.delphipraxis.net/internal...t.php?t=151373

du mußt ja kein Interface nehmen (TObject oder TBasis reichen auch)
Delphi-Quellcode:
Function TBasis.CreateInstance: TBasis; Virtual; Abstract;

Function TAbleitungA.CreateInstance: TBasis; Override;
  Begin
    Result := TAbleitungA.Create;
  End;
und das würde dann in jeder Ableitung entsprechend überschrieben.



was auch geht, wäre: (ist aber nicht grad flexibel )
Delphi-Quellcode:
if test.ClassType is TAbleitungA then kopie := TAbleitungA.Create
else if test.ClassType is TAbleitungB then kopie := TAbleitungB.Create
.....
[edit] da hat dächschen wohl schneller getippt
[edit2] abstract hatte ich es ... zumindestens im programm
$2B or not $2B
  Mit Zitat antworten Zitat
Dax
(Gast)

n/a Beiträge
 
#5

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 00:25
Dein zweiter Vorschlag ist ja wohl ein absolutes Antipattern, himi Und abstrakt sollte diese virtuelle Methode auf jeden Fall sein, sonst könnte jemand das nicht überschreiben und die Semantik der Methode stören.
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 00:31
Zitat von Dax:
Dein zweiter Vorschlag ist ja wohl ein absolutes Antipattern, himi
aber es funktioniert zumindestens

ok, abstract hatte ich vergessen zu schreiben
$2B or not $2B
  Mit Zitat antworten Zitat
Benutzerbild von stoxx
stoxx

Registriert seit: 13. Aug 2003
1.111 Beiträge
 
#7

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 02:01
Metaklassen sind Dein Stichwort ...

soo ...


Delphi-Quellcode:
type

TBaseClassType = class of TBasis;



var

 aBaseClass : TBaseClassType;
 aBaseObj : TBasis;
 
begin


  aBaseClass := TAbleitungA;
  
   // constructor create von TBASIS muss virtual deklariert sein (oder virutal; abstract)
   // constructor create von TAbleitungA muss als override; deklariert werden
  
  aBaseObj := aBaseClass.Create;
  Showmessage(aBaseObj.Classname);
  
  
  
end;
Phantasie ist etwas, was sich manche Leute gar nicht vorstellen können.
  Mit Zitat antworten Zitat
blackdrake

Registriert seit: 22. Aug 2003
Ort: Bammental
618 Beiträge
 
Delphi 10.3 Rio
 
#8

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 09:42
Hallo.

Vielen Dank für eure zahlreichen Antworten!

Sowohl die Meta-Klassen als auch die virtuell-abstrakten Clone-Methoden haben funktioniert:

Delphi-Quellcode:
type
  TBasis = class(TObject)
    Test: string;
    constructor Create;
    function Clone: TBasis; virtual; abstract;
  end;

  TAbleitung = class(TBasis)
    Test: string;
    constructor Create;
    function Clone: TBasis; override;
  end;

  TBaseClassType = class of TBasis;

{ TForm1 }

procedure TForm1.CloneVariant(Sender: TObject);
var
  Original, Kopie: TBasis;
  TempType: TBaseClassType;
begin
  Original:= TAbleitung.Create; // Dynamisch ermittelt, Derivat unbekannt
  Original.Test := 'Hubbi';

  TempType := TBaseClassType(Original.ClassType);

  Kopie := TBasis(TempType.Create);
  Kopie.Test := 'Flubby';

  showmessage(Original.Test);
  showmessage(Original.ClassName);
  showmessage(Kopie.Test);
  showmessage(Kopie.ClassName);

  // Ausgegeben:
  // Create Ableitung; Create Basis; Create Basis;
  // Hubbi; TAbleitung; Flubby; TAbleitung (OK)
end;

procedure TForm1.MetaClassVariant(Sender: TObject);
var
  Original, Kopie: TBasis;
begin
  Original := TAbleitung.Create; // Dynamisch ermittelt, Derivat unbekannt
  Original.Test := 'Hubbi';

  Kopie := Original.Clone;
  Kopie.Test := 'Flubby';

  showmessage(Original.Test);
  showmessage(Original.ClassName);
  showmessage(Kopie.Test);
  showmessage(Kopie.ClassName);

  // Ausgegeben:
  // Create Ableitung; Create Basis; Create Ableitung; Create Basis;
  // Hubbi; TAbleitung; Flubby; TAbleitung (OK)
end;

{ TBasis }

constructor TBasis.Create;
begin
  showmessage('Create Basis');
  inherited;
end;

{ TAbleitung }

constructor TAbleitung.Create;
begin
  showmessage('Create Ableitung');
  inherited;
end;

function TAbleitung.Clone: TBasis;
begin
  result := TAbleitung.Create;
  result.Test := Test;
end;
Ich denke, dass ich die Metaklassen verwenden werde, da dadurch mein Code aufgrund der nicht zu implementierenden Clone-Methoden etwas schlanker bleibt.

Ist das jetzt alles OK, was ich hier gemacht habe?

@stoxx: Mein Code funktioniert scheinbar, obwohl ich nicht folgende Hinweise beachtet habe:

Zitat:
// constructor create von TBASIS muss virtual deklariert sein (oder virutal; abstract)
// constructor create von TAbleitungA muss als override; deklariert werden
Wieso ist das so? Ist mein Code damit noch nicht in Ordnung und sollte ich etwas ändern?

Gruß
blackdrake
Daniel Marschall
  Mit Zitat antworten Zitat
Blup

Registriert seit: 7. Aug 2008
Ort: Brandenburg
1.477 Beiträge
 
Delphi 12 Athens
 
#9

Re: Zu unbekanntem Klassenderivat casten?

  Alt 13. Mär 2009, 16:48
Die Bezeichnungen für CloneVariant und MetaClassVariant sind vertauscht.

In deiner Variante ohne Clone wird der falsche Konstruktor nähmlich immer der von TBasis aufgerufen (weil nicht virtuell).
Außerdem werden die Variablen der Kopie nicht mit den Werten des Orginals vorbelegt.

Die Klasse Abteilung hat 2 Variablen mit dem Namen "Test". Eine geerbte von TBasis und eine eigene.


Hier noch ein Vorschlag für die Clone-Variante:
Delphi-Quellcode:
type
  TBasis = class(TObject)
    constructor Create; virtual;
  protected
    FName String;
  public
    property Name: string read FName write FName;
    procedure Assign(AObject: TBasis); virtual;
    function Clone: TBasis; virtual;
  end;

  TAbleitung = class(TBasis)
  protected
    FFunktion: String;
  public
    property Funktion: string read FFunktion write FFunktion;
    procedure Assign(AObject: TBasis); override;
  end;

  TSonderAbteilung = class(TAbleitung)
    constructor Create; override;
  protected
    FSonderfunktion: string;
  public
    property Sonderfunktion: String read FSonderfunktion write FSonderfunktion;
    procedure Assign(AObject: TBasis); override;
  end;

constructor TBasis.Create;
begin
  showmessage('constructor TBasis für ' + ClassName);
end;

procedure TBasis.Assign(AObject: TBasis);
begin
  FName := AObject.Name;
end;

function TBasis.Clone: TBasis;
begin
  Result := ClassType.Create;
  Result.Assign(Self);
end;

procedure TAbleitung.Assign(AObject: TBasis);
begin
  inherited;
  if AObject is TAbleitung then
    FFunktion := TAbleitung(AObject).Funktion;
end;

constructor TSonderAbteilung.Create;
begin
  inherited;
  showmessage('constructor TSonderAbteilung für ' + ClassName);
end;

procedure TSonderAbteilung.Assign(AObject: TBasis);
begin
  inherited;
  if AObject is TSonderAbteilung then
    FSonderfunktion := TSonderAbteilung(AObject).Sonderfunktion;
end;
  Mit Zitat antworten Zitat
blackdrake

Registriert seit: 22. Aug 2003
Ort: Bammental
618 Beiträge
 
Delphi 10.3 Rio
 
#10

Re: Zu unbekanntem Klassenderivat casten?

  Alt 15. Mär 2009, 20:50
Hallo.

Vielen Dank für die Antwort. Ich habe wie gesagt in diesem Fall jetzt die Metaklassen-Variante vorgezogen, weil ich mir hierdurch viel Codezeilen an Clone()-Funktionen spare.

Ich habe bei der Implementierung selbst herausgefunden, wieso das virtual+override doch wichtig war. Ich hatte 4 Varianten durchprobiert, die sich kompilieren liesen und auf dem ersten Blick zu funktionieren schienen:

1. Kopie := TBasis(Original.NewInstance);
2. Kopie := TBasis(Original.ClassType.Create);
3. Kopie := TBasis(Original.ClassType).Create;
4. Kopie := TBasis(Original.ClassType).Create; // + TBasis.Create als virtual;

Die Instanz wurde bei allen vier Programmen korrekt initialisiert und erhielt einen eigenen Speicherabschnitt.

Aber: Felder, die im Konstruktor deklariert waren, waren bei Variante 1-3 leer! Deswegen war noch Variante 4 die korrekte Lösung.

Hier noch mein kompletter Probecode:

Delphi-Quellcode:
type
  TBasisClass = class of TBasis;

  TBasis = class(TObject)
    F: String;
    Test: string;
    constructor Create; virtual;
  end;

  TAbleitung = class(TBasis)
    constructor Create; override;
  end;

procedure TForm1.Button2Click(Sender: TObject);
var
  Original, Kopie: TBasis;
begin
  Original := TAbleitung.Create;
  Original.Test := 'A';

  Kopie := TBasisClass(Original.ClassType).Create;
  Kopie.Test := 'B';

  showmessage('Feld Original: ' + Original.F); // "Ableitung"
  showmessage('Feld Kopie: ' + Kopie.F); // "Ableitung" (nur bei virtual constructor)
end;

constructor TBasis.Create;
begin
  inherited;
  F := 'Basis';
end;

constructor TAbleitung.Create;
begin
  inherited;
  F := 'Ableitung';
end;
Jetzt weiß ich auch endlich, was virtual bedeutet: Die Adresse der Methode wird zur Laufzeit ermittelt und nicht zur Compilerzeit. Das ist ja bei mir notwendig, da ich eine Instanz eines unbekannten Derivats erzeugen will. Man lernt nie aus.

Gruß
blackdrake
Daniel Marschall
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


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 22:22 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