Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   Delphi Auf private Variable zugreifen (https://www.delphipraxis.net/182774-auf-private-variable-zugreifen.html)

OlliWW 17. Nov 2014 14:01

Auf private Variable zugreifen
 
Hallo Zusammen,

Gibt es irgendeinen (schmutzigen) Trick um auf eine Private Variable zuzugreifen?

Ich habe folgende Deklaration:
Code:
TfrxBarcode2DView = class(TfrxView)
    private
        FBarCode: TfrxBarcode2DBase;
.....
Wenn ich jetzt ein Objekt der Klasse TfrxBarcode2DView erstelle müsste ich auf FBarCode zugreifen, leider ist die ja private und ich kann deren Wert nicht auslesen. Wie manche am Quellcode schon erkennen ist es der FastReport Sourcecode, weswegen Änderungen an der Deklaration an dieser Stelle für mich ausgeschlossen sind.
Gibt es irgendeinen Trick wie ich in meinen Units dennoch auf diese Variable zugreifen kann? Irgendwie überschreiben oder ähnliches vielleicht?

Andreas L. 17. Nov 2014 14:32

AW: Auf private Variable zugreifen
 
z. B. so:

Delphi-Quellcode:
  TfrxBarcode2DView = class(TfrxView)
  private
    FBarCode: TfrxBarcode2DBase;
  published
    property BarCode: TfrxBarcode2DBase read FBarCode;
    // wenn der Wert nicht nur ausgelesen sondern auch
    // verändert werden soll, dann kann oder muss noch ein Setter angegeben werden
    property BarCode: TfrxBarcode2DBase read FBarCode write SetBarCode;
    // Siehe ein Tutorial zu Klassen & deren Eigenschaften...
  end;

  ...

// Beispiel für den Zugriff/Aufruf:
procedure TForm1.Button1Click(Sender: TObject);
var
  2DView: TfrxBarcode2DView;
begin
  2DView := TfrxBarcode2DView.Create;
  try
    // Zugriff auf eine Funktion
    2DView.BarCode.FuncXYZ();
  finally
    FreeAndNil(2DView);
  end;
end;

DeddyH 17. Nov 2014 14:44

AW: Auf private Variable zugreifen
 
Wenn ich es richtig verstanden habe, kann man das Gewünschte damit aber nicht errreichen. Man hat zwar Zugriff auf FBarCode, aber der hat mit dem Feld in der Elternklasse nichts zu tun. :oops: Nicht richtig gelesen, das war ja eine Erweiterung der bestehenden Klasse und keine Ableitung.

himitsu 17. Nov 2014 14:55

AW: Auf private Variable zugreifen
 
Böse casten geht, aber man muß aufßassen, daß sich nicht verschiebt (beim nächsten Update).


Über die erweiterte RTTI erhält man auch Zugriff auf private Felder (solange das in der entsprechenden Unit/DLL nicht deaktiviert wurde).
Ob es hilbt ... k.A. da man die eigene DelphiVersion als Streng Geheim einstuft.

Zitat:

Wie manche am Quellcode schon erkennen ist es der FastReport Sourcecode, weswegen Änderungen an der Deklaration an dieser Stelle für mich ausgeschlossen sind.
Wieso?

Wir haben auch FastReport, inkl. der QuellCodes und da wir die Bibliotheken selber kompilieren, können wir da auch beliebig etwas ändern. (man muß nur beim Upgrade die Änderung wieder einbauen, falls der Bug nicht behoben wurde. :stupid:)

Daniel 17. Nov 2014 15:27

AW: Auf private Variable zugreifen
 
Ich löse sowas gern über einen Class-Helper. Zumindest mit XE7 kann man damit herrlich in fremden privaten Feldern herumwurschteln.
Die Code-Vervollständigung wird Dir die privaten Felder nicht anzeigen, der Compiler hingegen kommt gut damit klar.

Dejan Vu 17. Nov 2014 18:12

AW: Auf private Variable zugreifen
 
Na ja, was hier propagiert wird, ist ja eigentlich: "Scheiß auf OOP". Die Frage lautet ja eigentlich: Wieso meint der TE, an das private Feld zu müssen? Meistens geht es auch anders.

Der schöne Günther 17. Nov 2014 18:22

AW: Auf private Variable zugreifen
 
Zitat:

Zitat von Daniel (Beitrag 1280073)
Ich löse sowas gern über einen Class-Helper.

Das. In jeder Hinsicht besser als diese schrecklichen Interposer oder "Cracker"-Klassen.

Zitat:

Zitat von Dejan Vu (Beitrag 1280093)
Die Frage lautet ja eigentlich: Wieso meint der TE, an das private Feld zu müssen? Meistens geht es auch anders.

Die Antwort darauf hat er im letzten Absatz selbst gegeben.

Dejan Vu 17. Nov 2014 19:24

AW: Auf private Variable zugreifen
 
Die Antwort darauf hat er im letzten Absatz selbst gegeben.[/QUOTE]
Zitat:

Zitat von OlliWW (Beitrag 1280056)
Wenn ich jetzt ein Objekt der Klasse TfrxBarcode2DView erstelle müsste ich auf FBarCode zugreifen, leider ist die ja private und ich kann deren Wert nicht auslesen. Wie manche am Quellcode schon erkennen ist es der FastReport Sourcecode, weswegen Änderungen an der Deklaration an dieser Stelle für mich ausgeschlossen sind.
Gibt es irgendeinen Trick wie ich in meinen Units dennoch auf diese Variable zugreifen kann? Irgendwie überschreiben oder ähnliches vielleicht?

Wo steht jetzt genau, wieso er das machen will? Danke für die Nachhilfe.

Sherlock 18. Nov 2014 07:17

AW: Auf private Variable zugreifen
 
Das ist der Grund war ich private nicht mag, protected hat weitestgehend den selben Effekt, man darf die aber ererben.

Sherlock

DeddyH 18. Nov 2014 07:37

AW: Auf private Variable zugreifen
 
Das Feld selbst privat zu machen ist IMO durchaus in Ordnung, man kann sich aber überlegen, Getter und Setter unter protected zu deklarieren.

JasonDX 18. Nov 2014 08:05

AW: Auf private Variable zugreifen
 
Ob/Wann man private verwendet hängt davon ab, welche Garantien man für dieses Feld geben und erwarten will. Wenn ich private member deklariere, dann in der Annahme, dass meine Klasse, und NUR meine Klasse diese liest und bearbeitet. Wenn ich also einen privaten String phoneNumber habe, und den in meiner Klasse nur mit einem bestimmten Format fülle, gehe ich im Rest der Klasse davon aus, dass dieser String ein bestimmtes Format hat.
Wenn nun andere Klassen diesen Wert schreiben, habe ich diese Garantie nicht mehr. Deswegen: Schreibe nie private Member von anderen Klassen. Die Klasse selbst erwartet u.U. bestimmte Konventionen, und hat ein undefiniertes Verhalten wenn diese Konventionen nicht eingehalten werden.
Das selbe geht in die andere Richtung: In einem kleinen Update ändere ich das Format dieses Strings. Das Update kann schleichend kommen - es beinhaltet keine Veränderung des APIs, und interessiert auch keinen außerhalb. Es ist schließlich das persönliche private member der Klasse. Wenn nun aber eine andere Klasse diesen Wert liest, und annimmt, dass der String ein bestimmtes Format hat, bringt diese Änderung ein undefiniertes Verhalten hervor, weil Annahmen, die für dieses Feld getroffen wurden, nicht mehr gelten.

Wenn man bspw. einen von außerhalb zugänglichen Getter setzt (sei es durch Private und Kindklassen), dann setzt man bestimmte Garantien für das Feld, die ein einfaches Ändern nicht mehr erlauben, bzw. der Getter entsprechende Umwandlung o.ä. vollzieht. Er ist aber Teil des APIs und muss damit den dokumentierten Anforderungen genügen. Ebenso andersrum: hat man einen von außerhalb zugänglichen Setter, können dafür Erwartungen definiert werden, die evt. auch im Setter überprüft werden (bspw. dass der String in einem bestimmten Format sein soll)
Der wichtigste Punkt ist: Diese Erwartungen und Garatien sind dann dokumentiert, und können(/sollten) sich nicht ohne weiteres von einen Tag auf den nächsten ohne Ankündigung ändern. Das gilt nicht für private member.

Kurzum: Private hat seine Daseinsberechtigung, und sollte auch nicht durch Tricks/Hacks umgangen werden. Es endet in einem Haufen Dung in der Codebase.

Elrond 18. Nov 2014 08:19

AW: Auf private Variable zugreifen
 
Ich weiß nicht inwiefern das jetzt passt.
Aber ich musste auch mal auf ein privat Feld zugreifen und zwar wollte ich damals von TField den TFieldType zur Laufzeit ändern ohne es in jeden Formular anzupassen. Das ganze habe ich über die RTTI realisiert, dass klappt soweit ich weiß nur bei private Feldern die nicht strict private sind.

Delphi-Quellcode:
procedure ChangeProperty(Sender: TObject);
  var
    Context                            : TRttiContext;
    RttiField                          : TRttiField;
    RttiValue                          : TValue;
  begin
    RttiField := Context.GetType(TField).GetField('FDataType');
    RttiValue := TValue.FromOrdinal(TypeInfo(TFieldType), 24); //24 ftWideString

    RttiField.SetValue(Sender, RttiValue);
  end;


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