Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Inherited "überspringen"? (https://www.delphipraxis.net/192045-inherited-ueberspringen.html)

Rabenrecht 16. Mär 2017 13:40

Delphi-Version: 5

Inherited "überspringen"?
 
Man betrachte folgende Struktur:

Delphi-Quellcode:
  Oberklasse = class(TObject)
  protected
    procedure Proc; virtual;

  Unterklasse1 = class(Oberklasse)
  protected
    procedure Proc; override;

  Unterklasse2 = class(Unterklasse1)
  protected
    procedure Proc; override;
Man nehme weiterhin an, dass ich als Programmierer nur den Code ab Unterklasse2 modifizieren kann (weil zb. der Rest in irgendwelchen Standardbibliotheken ist).

Zudem sei Unterklasse1.Proc so implementiert, dass es in Randfällen im Anwendungskontext, in dem auch Unterklasse2 steht (also das Programm, von dem Unterklasse2 ein Teil ist), zu Fehlern kommt.
Dies ist überhaupt erst der Grund, warum ich in Unterklasse2 Proc nocheinmal implementiere.

Nun ist es so, dass in Unterklasse1.Proc inherited, also Oberklasse.Proc aufgerufen wird. Diesen Aufruf würde ich auch gerne behalten.

Wenn ich aber Unterklasse2.Proc folgendermaßen implementiere:
Delphi-Quellcode:
procedure Unterklasse2.Proc;
begin
  [...]
  inherited;
end;
würde der (fehlerhafte) Code von Unterklasse1.Proc wieder aufgerufen werden.

Lange Rede, kurze Frage: ist es möglich beim inherited Stufen in der Hierarchie zu "überspringen"? Oder eine spezifische Implementierung von Proc aufzurufen?

Ghostwalker 16. Mär 2017 14:00

AW: Inherited "überspringen"?
 
Naja....du könntest z.B. folgendes machen:

Delphi-Quellcode:
procedure Unterklasse2.Proc;
begin
  [...]
  Oberklasse.proc;
end;

Rabenrecht 16. Mär 2017 14:02

AW: Inherited "überspringen"?
 
Hm... Tatsache. Tja, manchmal geht's es ganz einfach :-D
Danke!

Uwe Raabe 16. Mär 2017 14:07

AW: Inherited "überspringen"?
 
Zitat:

Zitat von Ghostwalker (Beitrag 1364401)
Delphi-Quellcode:
procedure Unterklasse2.Proc;
begin
  [...]
  Oberklasse.proc;
end;

Das hatte ich auch erst gedacht, aber weder unter Berlin noch XE2 compiliert das überhaupt:

Zitat:

[DCC Fehler]E2076 Diese Form des Methodenaufrufs ist nur für Klassenmethoden erlaubt
Funktioniert das denn bei dir?

hoika 16. Mär 2017 14:11

AW: Inherited "überspringen"?
 
Hallo,
ist es möglich beim inherited Stufen in der Hierarchie zu "überspringen"? Oder eine spezifische Implementierung von Proc aufzurufen?

Kurze Antwort: nein
Lange Antwort: es kommt drauf an

In deiner eigenen Klasse kannst du das inherited ja weglassen,
musst dann aber wissen, was in den darüberliegenden Klasse gemacht wird,
dann könntest du aber gleich reintroduce nehmen.
Oder du rufst das inherited auf und korrigierst dann die falschen Daten,
falls das geht.

bra 16. Mär 2017 14:26

AW: Inherited "überspringen"?
 
Eventuell so?

Delphi-Quellcode:
procedure Unterklasse2.Proc;
begin
  [...]
  Oberklasse(Self).proc;
end;

SProske 16. Mär 2017 14:30

AW: Inherited "überspringen"?
 
Zitat:

Zitat von bra (Beitrag 1364413)
Eventuell so?

Delphi-Quellcode:
procedure Unterklasse2.Proc;
begin
  [...]
  Oberklasse(Self).proc;
end;

Das funktioniert nur solange beide Klassen in der selben Unit sind und Proc protected, aber nicht strict protected ist. Für den beschriebenen Anwendungsfall also ungeeignet.

hoika 16. Mär 2017 14:37

AW: Inherited "überspringen"?
 
Hallo,

Delphi-Quellcode:
Oberklasse(Self).proc;
Wenn aber Oberklasse.Proc dann das fehlerhafte inherited aufruft,
geht wieder alles schief ...

bra 16. Mär 2017 14:40

AW: Inherited "überspringen"?
 
Zitat:

Zitat von hoika (Beitrag 1364416)
Hallo,

Delphi-Quellcode:
Oberklasse(Self).proc;
Wenn aber Oberklasse.Proc dann das fehlerhafte inherited aufruft,
geht wieder alles schief ...

Oberklasse ist im Beispiel aber das auf der ersten Ebene, hat also kein kritisches inherited.

Uwe Raabe 16. Mär 2017 15:02

AW: Inherited "überspringen"?
 
Zitat:

Zitat von hoika (Beitrag 1364416)
Delphi-Quellcode:
Oberklasse(Self).proc;

Das geht in jedem Fall schief, da du damit auch nur die virtuelle Methode, also wieder Unterklasse2.Proc, aufrufst und so in einen Stack-Overflow läufst.

Die einzige Möglichkeit, die mir dazu einfällt ist schon arg bösartig:

Delphi-Quellcode:
procedure Unterklasse2.Proc;
begin
  asm
    mov eax,self
    call Oberklasse.Proc
  end;
//  inherited;
  Writeln('Unterklasse2');
end;

jaenicke 16. Mär 2017 15:51

AW: Inherited "überspringen"?
 
Ich würde das lieber so sauber wie eben möglich ohne Assembler lösen:
Delphi-Quellcode:
uses
  System.Rtti, System.TypInfo;

procedure Unterklasse2.Proc;
begin
  System.Rtti.Invoke(@Oberklasse.Proc, TArray<TValue>.Create(TValue.From<Unterklasse2>(Self)), ccReg, nil);
end;

Ghostwalker 16. Mär 2017 18:11

AW: Inherited "überspringen"?
 
Außer den Möglichkeiten von Uwe und Jaenike hab ich keine weiteren gefunden.

Da du aber Unterklasse2 eh nur wegen eines Fehlers in Unterklasse1 baust, warum
leitest du das ganze nicht direkt von Oberklasse ab ? Dann kann Unterklasse1 über die
Klinge springen.

Rabenrecht 17. Mär 2017 12:36

AW: Inherited "überspringen"?
 
Zitat:

Zitat von Uwe Raabe (Beitrag 1364405)

Funktioniert das denn bei dir?

Ne...

Mein spezifisches Problem war anders lösbar, weil die Oberklasse.Proc nen Einzeiler mit nem Zugriff auf eine Published Property war (check auf Ereignisbehandlung) und ich das deshalb schmerzfrei nachbauen konnte.

Die Diskussion interessiert mich dennoch. Ähnliche Fälle werden in Zukunft wahrscheinlich nochmal auftreten.


Zitat:

Zitat von Ghostwalker (Beitrag 1364430)
Da du aber Unterklasse2 eh nur wegen eines Fehlers in Unterklasse1 baust, warum
leitest du das ganze nicht direkt von Oberklasse ab ? Dann kann Unterklasse1 über die
Klinge springen.

Weil im Unterschied zu meinem aufs Notwendigste reduzierte Beispiel im konkreten Fall Unterklasse1 eine ganze Menge notwendiger Funktionalität enthält :wink:

Zacherl 17. Mär 2017 13:28

AW: Inherited "überspringen"?
 
Noch eine Dritte Variante ohne Assembly oder RTTI:
Delphi-Quellcode:
{ TA }

procedure TA.X(I: Integer);
begin
  ShowMessage('1: ' + I.ToString);
end;

{ TB }

procedure TB.X(I: Integer);
begin
  inherited;
  ShowMessage('2: ' + I.ToString);
end;

{ TC }

procedure TC.X(I: Integer);
type
  TProc = procedure(Sender: TA; I: Integer);
begin
  //inherited;
  TProc(@TA.X)(Self, I);
  ShowMessage('3: ' + I.ToString);
end;

MichaelT 17. Mär 2017 17:19

AW: Inherited "überspringen"?
 
Nein. Solltest du meiden.

Es gibt in C++ diese Art des Aufrufs, mit dem wird aber im Falle von Multiple Inheritance zwischen den Base Classes unterschieden in erster Linie.

Gute Frage wäre ob du nicht einfach ein Objekt der entfernten Basisklasse erzeugst und das Ergebnis verwendest. Ob das Sinn macht sei dahingestellt.

Ich würde nicht ausschließen, ...


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