Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Algorithmen, Datenstrukturen und Klassendesign (https://www.delphipraxis.net/78-algorithmen-datenstrukturen-und-klassendesign/)
-   -   Delphi Problem mit virtuellen Methodenaufrufen im Constructor (https://www.delphipraxis.net/172729-problem-mit-virtuellen-methodenaufrufen-im-constructor.html)

Zacherl 19. Jan 2013 14:26

Problem mit virtuellen Methodenaufrufen im Constructor
 
Hallo zusammen,

ich habe ein schwerwiegendes Problem in meinem aktuellen Klassendesign. Meine Komponenten erhalten von sich selbst oder von anderen Komponenten Nachrichten (keine Window Messages!), welche dann in virtuellen Methoden abgearbeitet werden.

Nun habe ich folgendes Problem:
1) Basisklasse führt im Constructor eine Operation aus, die dazu führt, dass eine der virtuellen Methoden aufgerufen wird
2) abgeleitete Klasse hat diese Methode überschrieben und greift z.b. auf eine Liste zu
3) die Liste kann allerdings erst nach Aufruf des originalen Constructors initialisiert werden und ist somit zum Zeitpunkt der Nachricht undefiniert
4) Access Violation

Gut, hier könnte ich einfach auf nil prüfen, doch bei anderen Aktionen würde ich durch "ignorieren" der Nachricht weitere Folgefehler auslösen.

Hat hier jemand einen Tipp für mich? Habe gesehen, dass es Methoden wie AfterConstruction gibt. Aber können die mir hier helfen?

Viele Grüße
Zacherl

sahimba 19. Jan 2013 14:47

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Zitat:

Zitat von Zacherl (Beitrag 1199767)
Habe gesehen, dass es Methoden wie AfterConstruction gibt. Aber können die mir hier helfen?

Durchaus, da dann alle Konstruktoren "durchgelaufen" sind. Verschiebe also den Aufruf der virtuellen Methode dorthin.

Grüße,
Stefan

Furtbichler 19. Jan 2013 15:54

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Ein Konstruktor sollte nie eine Aktion auslösen, sondern nur und ausschließlich Datenstrukturen initialisieren. Befolge diese Grundregel und Du hast weniger Probleme (programmiertechnisch).

Zacherl 20. Jan 2013 23:57

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Danke euch beiden. Konnte mein Problem durch Überschreiben der AfterConstruction() Methode lösen :thumb:

Furtbichler 21. Jan 2013 06:27

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Hi,

dein lokales Problem mag damit gelöst sein, aber trotzdem schaffst Du es nun, das deine Klasse schon beim Aufruf des Konstruktors eine Exception wirft (natürlich kann das immer passieren, EOutOfMemoryException z.B.). Dein Konstruktoraufruf ist ja i.a. außerhalb des Try-Finally und wird somit nicht ordentlich abgefangen.

Ich würde mir überlegen, ob deine Klassenfamilie nicht mit einer Methode à la 'Initialize' besser bedient wäre. Denn die könnte innerhalb des Try-Finally aufgerufen werden.

Meine persönliche Meinung, is klar.

sahimba 21. Jan 2013 10:05

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Siehe auch (wenngleich .NET)
http://msdn.microsoft.com/de-de/libr.../ms182331.aspx

Zacherl 21. Jan 2013 12:51

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Zitat:

Zitat von Furtbichler (Beitrag 1199933)
Hi, dein lokales Problem mag damit gelöst sein, aber trotzdem schaffst Du es nun, das deine Klasse schon beim Aufruf des Konstruktors eine Exception wirft (natürlich kann das immer passieren, EOutOfMemoryException z.B.). Dein Konstruktoraufruf ist ja i.a. außerhalb des Try-Finally und wird somit nicht ordentlich abgefangen.

Ich bin selbst absolut kein Fan von Exceptions, die im Konstructor ausgelöst werden (TFileStream und Konsorten), in diesem Falle ist es vom logischen Konzept her aber nicht besser machbar. Meine Klasse erwartet im Konstruktor einen Parent, bei dem sie sich dann als Child Komponente registriert. Ist dieser Parent Wert nicht gesetzt, muss ich wohl oder übel eine Exception schmeißen. Die anderen Aktionen, die ich vorher unnötigerweise im Konstruktor ausgeführt habe, konnte ich jetzt aber in dafür vorgesehene Methoden auslagern :)

Medium 21. Jan 2013 14:03

AW: Problem mit virtuellen Methodenaufrufen im Constructor
 
Das müsste sich doch ebenfalls über eine Methode SetParent(), oder gar eine hübsche Property mit Setter machen lassen - oder übersehe ich hier etwas?

Edit: Etwas von hinten durch die Brust, aber ggf. auch ein Weg: Als letztes im Konstruktor eine User-Message mit Pointer auf den Parent an den Aufrufer posten (für das Handle ließe sich ja ggf. auch der Parent-Parameter missbrauchen), und in diesem dann auf die Message mit diesem SetParent reagiern. Dann löst doch der Konstruktor das Setzen des Parents aus, aber tut es nicht wirklich. Etwas seltsam, gebe ich zu, aber wenn es denn sein muss :)
Wenn das Vorhandensein des Parents nur für weitere Operationen mit der Klasse nötig ist, und im Konstruktor eigentlich funktional nicht genutzt wird, wäre es generell aber besser, die Methoden so zu gestalten, dass DIESE eine Exception werfen, wenn der Parent nicht gültig ist. Dann gib't die Exception nur bei der verwendung, nicht beim Erzeugen. Besser finde ich.


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