![]() |
Korrekte virtuelle Methode(n) in abgel. Klasse aufrufen?
Hallo,
um die Aufgabenstellung zu präzisieren, die uns unser Lehrer aufgegeben hat will ich mal eine Klassenhirarchie bauen, die mein Problem verdeutlichen soll:
Delphi-Quellcode:
Einer meiner Mitschüler hat ein größeres Projekt in Arbeit, wo der Aufruf nicht korrekt zu sein scheint. Wie kann ich nun auch in großen unübersichtlichen Projekten sicher stellen, das immer die richtige virtuelle Methode aufgerufen wird, in meinem Beispiel die Methode "DoIt".
unit ClassesWithVirtualMethods;
interface type TMyBaseClass = class(TComponent) procedure DoSomething; procedure DoIt; virtual; end; TMyDerivedClass = class(TMyBaseClass) procedure DoIt; override; end; TMySecondDerivedClass = class(TMyDerivedClass) procedure DoIt; override; end; TCallerClass = class(TMyBaseClass) procedure Call; end; implementation procedure TMyBaseClass.DoIt; begin Show('TMyBaseClass'); end; procedure TMyDerivedClass.DoIt; begin Show('TMyDerivedClass'); end; procedure TMySecondDerivedClass.DoIt; begin Show('TMySecondDerivedClass'); end; procedure TCallerClass.Call; var index: Integer; begin //Annahme: Es seien mehrere Komponenten im Projekt enthalten. REPEAT self.DoIt; //self nur zur Verdeutlichung. Ich weiß, das der self Parameter hier weggelassen wird for Index:=0 to self.ComponentCount-1 do begin if Assigned(TMySecondDerivedClass(Components[Index])) then TMySecondDerivedClass(Components[Index]).DoIt; end; UNTIL endebedingung; //Pseudocode end; end. Da das Projekt meines Mitschülers in das Themengebiet meines derzeitigen Lehrstoffes passt, nämlich virtuelle Methoden, Polymorphie... , stelle ich diese Frage mal hier, in der Hoffnung, das mir jemand Rat geben kann, wie ich sicher stelle, das die richtige DoIt Methode aufgerufen wird. Auf unserem Windows Desktop wäre das ja entweder eine Methode des Desktop oder die der Hapuptform oder die einer Eingabekomponente auf dieser Hauptform. Wie kann ich sicher stellen, das zu jeder Zeit die richtige Methode (der richtigen Klasse), wo zum Beispiel gerade der Mauszeiger ist, aufgerufen wird? |
Re: Korrekte virtuelle Methode(n) in abgel. Klasse aufrufen?
Ich habe ehrlich gesagt nicht ganz verstanden, worauf du hinaus willst. Was ist denn die "richtige" Methode?
Überprüfen ob ein Objekt einer bestimmten Klasse entspricht kannst du mir dem Operator ![]() |
Re: Korrekte virtuelle Methode(n) in abgel. Klasse aufrufen?
Alle Möglichkeiten, welche man offiziell und ohne wildes Rumgepointere zur Verfügung hat
Delphi-Quellcode:
TMyBaseClass = class(TComponent)
procedure DoIt; virtual; end; TMyDerivedClass = class(TMyBaseClass) procedure DoIt; override; end; TMyDerivedClass2 = class(TMyBaseClass) procedure DoIt; override; // no inherited procedure DoIt_Prev; end; procedure TMyBaseClass.DoIt; begin Show('TMyBaseClass'); end; procedure TMyDerivedClass.DoIt; begin Show('TMyDerivedClass'); Inherited; end; procedure TMyDerivedClass2.DoIt; begin Show('TMyDerivedClass 2 no prev'); end; procedure TMyDerivedClass2.DoIt_Prev; begin Inherited; end;
Delphi-Quellcode:
var Test: TMyBaseClass;
begin Test := TMyBaseClass.Create; Test.DoIt; Test.Free; Test := TMyDerivedClass.Create; Test.DoIt; Test.Free; Test := TMyDerivedClass2.Create; Test.DoIt; Test.DoIt_Prev; Test.Free; end; |
Re: Korrekte virtuelle Methode(n) in abgel. Klasse aufrufen?
Zitat:
das ist doch eine Tautologie - der Zweck virtueller Methoden ist es, Klassen so entwerfen zu können, dass immer die richtige Methode aufgerufen wird. Hält man sich daran, dann ist die automatisch aufgerufene Methode die Richtige - schon per definitionem. Wird eine andere Methode verwendet als du meinst, dann ist die Klasse falsch entworfen oder codiert. Gruss Reinhard |
Re: Korrekte virtuelle Methode(n) in abgel. Klasse aufrufen?
Die Methode Call sollte so aussehen:
Delphi-Quellcode:
Die Hilfsvariable x ist vom Datentyp der Klasse, in der du die virtuelle Methode DoIt() eingeführt hast.
procedure TCallerClass.Call;
var index: Integer; x : TMyBaseClass; // Hilfsvariable begin //Annahme: Es seien mehrere Komponenten im Projekt enthalten. REPEAT self.DoIt; //self nur zur Verdeutlichung. Ich weiß, das der self Parameter hier weggelassen wird for Index:=0 to self.ComponentCount-1 do begin x := Components[Index] as TMyBaseClass; if Assigned(x) then x.DoIt; end; UNTIL endebedingung; //Pseudocode end; Achja: die Hilfsvariable erleichtet uns das Leben, denn man braucht nur einmal zu Casten und hat dann auch im Debugger den Vorteil mehr zu sehen als ohne die Hilfsvariable. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 02: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-2025 by Thomas Breitkreuz