![]() |
TObjectList-Object da, aber beim Zugriff Stack-Overflow
Hi,
ich habe eine TObjectList nach diesem Schema:
Delphi-Quellcode:
Explizit geht es um die Funktion:
type
TTemperatur=class private FTemp: Extended; FId: string; procedure SetId(const Value: string); procedure SetTemp(const Value: Extended); public constructor Create; procedure WriteToStream(st: TStream); procedure LoadFromStream(st: TStream); published property Id: string read FId write SetId; property Temp: Extended read FTemp write SetTemp; end; type TTemperaturList=class(TObjectList) function getItem(index: Integer): TTemperatur; virtual; procedure setItem(index: Integer; Temperatur: TTemperatur); virtual; public property Items[index: Integer]: TTemperatur read getItem write setItem; default; procedure Insert(index: Integer; Temperatur: TTemperatur); virtual; function Add(Temperatur: TTemperatur): Integer; virtual; function Remove(Temperatur: TTemperatur): Integer; virtual; function IndexOf(Temperatur: TTemperatur): Integer; virtual; function First: TTemperatur; virtual; function Last: TTemperatur; virtual; procedure WriteToStream(st: TStream); procedure LoadFromStream(st: TStream); function IndexOfId(value: string): Integer; virtual; end; type TRaspberry=class private FName: string; FTemp: TTemperaturList; FIP: string; procedure SetIP(const Value: string); procedure SetName(const Value: string); procedure SetTemp(const Value: TTemperaturList); public constructor Create; destructor Destroy; procedure WriteToStream(st: TStream); procedure LoadFromStream(st: TStream); published property Name: string read FName write SetName; property IP: string read FIP write SetIP; property Temp: TTemperaturList read FTemp write SetTemp; end; type TRaspberryList=class(TObjectList) function getItem(index: Integer): TRaspberry; virtual; procedure setItem(index: Integer; Raspberry: TRaspberry); virtual; public property Items[index: Integer]: TRaspberry read getItem write setItem; default; procedure Insert(index: Integer; Raspberry: TRaspberry); virtual; function Add(Raspberry: TRaspberry): Integer; virtual; function Remove(Raspberry: TRaspberry): Integer; virtual; function IndexOf(Raspberry: TRaspberry): Integer; virtual; function First: TRaspberry; virtual; function Last: TRaspberry; virtual; procedure WriteToFile(Filename: string); procedure LoadFromFile(Filename: string); function IndexOfIP(value: string): Integer; virtual; end;
Delphi-Quellcode:
der TTemperaturList.
function TTemperaturList.IndexOfId(value: string): Integer;
var I: Integer; begin Result:=-1; for I := 0 to self.Count-1 do if self[I].Id=value then begin Result:=I; Exit; end; end; Jetzt sollen mit folgender Prozedur TTemperatur-Items hinzugefügt, bzw. wenn sie schon vorhanden sind, geändert werden:
Delphi-Quellcode:
Ein TRaspberry wird manuell erzeugt, ist also vorhanden.
procedure TfrmMain.btn1Click(Sender: TObject);
var r, t: integer; Raspberry: TRaspberry; Temp: TTemperatur; FIP, FId: string; FTemperatur: Extended; begin FIP:='192.168.0.110'; FId:='28-000444e3ba3ff'; FTemperatur:=19.937; r:=FRaspberrys.IndexOfIP(FIP); if r>-1 then begin Raspberry:=FRaspberrys[r]; end else begin Raspberry:=TRaspberry.Create; FRaspberrys.Add(Raspberry); end; if Raspberry<>nil then begin t:=Raspberry.Temp.IndexOfId(FId); //<---Hier tritt beim zweiten durchlauf dann die Exception auf. if t>-1 then begin Temp:=Raspberry.Temp[t]; end else begin Temp:=TTemperatur.Create; Raspberry.Temp.Add(Temp); end; if Temp<>nil then begin Temp.Id:=FId; Temp.Temp:=FTemperatur; end; end; end; Beim TTemperaturList.IndexOfId(value: string) kommt es beim zweiten Aufruf der btn1Click zu einem Stackoverflow. Klar beim ersten mal wird das neue Objekt TTemperatur angelegt, weil eben noch keines vorhanden ist. Beim zweiten Mal müsste er es ja finden, da die Id die gleich ist. Komischerweise wird in der IndexOfId als Count auch 1 ausgegeben, aber auf das Object kann ich nicht zugreifen. Also irgendwo hab ich einen Denkfehler, aber ich finde ich einfach nicht. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Zitat:
Robuster wäre es die Containerklasse nicht von der Klasse TObjectList abzuleiten sondern eine interne (private) TObjecList Instanz zu benutzen. Das entkoppelt die API der TObjectList und sichert den Client-Code besser gegen "versehentliche" Verwendung von TObjectList Methoden ab. Das könnte auch für die konkrete Fehlersuche hilfreich sein. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Danke für deine Hilfe. Den eigentlichen Fehler habe ich mittlerweile gefunden: es war lediglich ein fehlendes inherited beim Zugriff auf das TTemperaturobjekt.
Zitat:
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Bei mir sind Container immer z.B. so aufgebaut:
Delphi-Quellcode:
Von aussen hin kann ich dann gezielt nur die von meiner Klasse definierten Funktionen und Prozeduren aufrufen. Dass innen eine TObjectList oder etwas anderes enthalten ist, soll der Client nicht sehen können.
type
TTemperaturList=class(TObject) private Liste: TObjectList; ... public constructor Create; destructor Destroy; override; |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Ich denke er meint, dass du eine Objektlist in deiner neuen Klasse kapselst. So kann man ganz gut die eventuelle Object-Casterei verbergen.
Edit:Und war sogar selber schneller als ich...
Delphi-Quellcode:
type
TTemperaturList=class ObjectList:TObjectList; public constructor create; function count:integer; end; TTemperaturList.create; begin ObjectList:=TObjectList.create; end; TTemperaturList.count:integer; begin Result:=ObjectList.Count; end; Andere Bemerkung, wenns erlaubt ist: Was mich an deinem Klassenkonstrukt stört ist was für verschiedene Dinge nicht alles Temp heißen. Da hat ich nämlich auch Probleme deinen QT nachzuvollziehen. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Delphi-Quellcode:
Die Methoden getItem und setItem brauchen nicht als published declariert werden, eventuell fehlt hier das private. Published wird nicht benötigt ausser man möchte die Klasse über das Delphi DFM Streaming oder klassische RTTI ansprechen.
type
TTemperaturList=class(TObjectList) function getItem(index: Integer): TTemperatur; virtual; procedure setItem(index: Integer; Temperatur: TTemperatur); virtual; public property Items[index: Integer]: TTemperatur read getItem write setItem; default; procedure Insert(index: Integer; Temperatur: TTemperatur); virtual; function Add(Temperatur: TTemperatur): Integer; virtual; function Remove(Temperatur: TTemperatur): Integer; virtual; ... |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Da der TE wohl XE4 verwendet, wäre sicher auch die generische Objektliste aus der Unit System.Generics.Collections einen Blick Wert.
Damit könnte man sich die Schreibarbeit für Add/Remove/IndexOf etc. ersparen. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
@mjustin:
das Private fehlt tatsächlich. Eigentlich kommen bei mir die Getter und Setter immer in Private, hatte ich lediglich vergessen. (hätte bei dem fehlenden Inherited aber nichts geändert). Aber ich weiß jetzt was du meinst. Klar kann man das auch über Generics verwenden. Aber ich habe mich so an meine ObjectList gewöhnt.... Und, zugegeben, der Variablenname Temp ist sicherlich nicht der schlaueste, aber man kann das schon lesen ;) |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Nur so als Tip:
Wenn man Klassen mit Stream-Funktionen ausstattet, dann sollte man auch immer das Interface ![]() Macht auf jeden Fall das Leben leichter ;) |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Zitat:
![]() |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Zitat:
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Hi Benmik,
ich habe mal ein kleines Einsteigertutorial für Interfaces erstellt. Vielleicht hilft es Dir ja etwas: ![]() Interfaces sind hilfreich, wenn man gleiche Funktionalitäten in unterschiedlichen Klassen unterbringen will. Es erspart einem das Prüfen und Casten von vorliegenden Objekten wie
Delphi-Quellcode:
Statt dessen schreibt man einfach
if (MyObj is ClassA) then
(MyObj as ClassA).DoX else if (MyObj is ClassB) then (MyObj as ClassB).DoX;
Delphi-Quellcode:
Dan ist völlig egal, was da für ein Objekt dahinter steckt. Wichtig ist nur noch, ob DoX unterstützt wird oder nicht.
var
lDoX: IDoX; ... if Supports(MyIntf, IDoX, lDoX) lDoX.DoX; Fluch und Segen (und außerhalb von Delphi unüblich) ist die automatische Referenzzählung. Objekte hinter Schnittstellen werden automatisch freigegeben, wenn es keine Referenzen mehr darauf gibt. Das kann man als positiv oder auch negativ bewerten - je nach den vorliegenden Bedürfnissen. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Hallo Stahli, deine Produktionen waren für mich sozusagen erste Anlaufstelle. Ich habe zum ersten Mal im Groben kapiert, was Interfaces überhaupt sind. Jetzt stehe ich vor der moralischen Entscheidung :? : Muss ich die auch verwenden? Die meisten Tutorial-Autoren machen eine Bemerkung, dass für "einfache" Delphi-Programme Interfaces meist nicht notwendig sind. Deshalb hat mich Sir Rufos Bemerkung verwirrt: Warum sollte mann immer das Interface "anheften"? Ist das wirklich die Regel, dass man so viele Objekte hat, die man unter einem Interface versammeln muss?
Und was ich über die Tücken der Referenzzählung gelesen habe: Da sträuben sich die Haare. Also, selbst wenn ich drei Objekte hätte, ich würde die Klassen einzeln erstellen und gut ist. Voraussetzung natürlich: Ich habe weder Schnittstellen nach außen noch irgendwelche Plugins oder Ähnliches. Aber vielleicht übersehe ich da was. Daher die Bitte um eine praktische Verwendung. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Zitat:
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Und könntest du neben Stahli auch mich mit einer Antwort bedenken?
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Zitat:
Habe ich damals beim designen der Klasse nachgedacht und das Interface
Delphi-Quellcode:
implementiert, dann sieht mein Code so aus
IStreamPersist
Delphi-Quellcode:
Und wenn ich es wieder zurück haben möchte?
MyDataSet.Append;
MyDataSet.FieldByName( 'BlobData' ).Assign( MyInstance ); MyDataSet.Post;
Delphi-Quellcode:
// leider nur sehr kurz
MyInstance.Assign( MyDataSet.FieldByName( 'BlobData' ) ); |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Hab Dank. Das muss ich mir erstmal zu Gemüte führen. Vermutlich liegt für mich der Punkt, an dem ich mal Interfaces brauchen kann, in einiger Ferne. Das war bei generischen Objektlisten allerdings auch mal so, und heute kann ich mir ein Leben ohne sie gar nicht mehr vorstellen. :thumb:
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
@Benmik
Also grundsätzlich würde ich bei einem überschaubaren Projekt keine Interfaces verwenden. Erst mal bringt das Dir nämlich nichts. Interfaces bringen Dir nur etwas, wenn sie Vorteile bringen. ;-) Wenn die Wahrscheinlichkeit besteht, dass Du später verschiedene Klassen dynamisch tauschen möchtest, dann machen Interfaces (und der etwas höhere Aufwand) Sinn. Man kann dann eher in "Funtionalitäten" denken und die Klassen sind dann "Funktionalitätenblöcke" (IkannA, IkannB, IkannC). Wenn Du mit Klassen nicht mehr weiter kommst und andauernd casten musst, dann denke halt über den Einsatz von Interfaces nach. Als reinen Selbstzweck macht das wenig Sinn. @Sir Rufo Oh, ich sehe gerade, man kann Interfaces offenbar auch TObject zuweisen und nicht nur TInterfacedObject und somit ohne Referenzzählung arbeiten. War mir noch gar nicht klar. Werde ich heute Abend gleich mal versuchen. Ich dachte es geht nur mit (Leer-)Überschreiben der Referenzzählung. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Mit einem Interface legt man fest, was die Klasse implementieren muss.
Basis für jedes Interface ist dabei
Delphi-Quellcode:
und das legt 3 Methoden fest, die also immer wie auch immer vorhanden sein müssen.
IInterface
Wie diese Implementierung und das Verhalten aussieht ... bleibt dem Klassen-Designer überlassen.
Delphi-Quellcode:
hat diese 3 Methoden schon implementiert und ich kann von da meine Klasse ableiten ... wenn die Implementierung von
TInterfacedObject
Delphi-Quellcode:
auch passt.
TInterfacedObject
Und es gibt noch eine Menge mehr Spielarten gerade im Bezug auf Interfaces, die das ganze dann auch sehr flexibel machen. |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Stimmt, ich hatte mich hier von einem anderen Thread in die Irre führen lassen. Aber Das Überschreiben (Stillegen) der Referenzzählung werde ich nochmal versuchen.
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Zitat:
![]() |
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
Naja, die Delphi-Gurus drohen mit Exkommunikation, wenn man sich nicht an ihre Bibeln hält (Nick Hodges: "If I could teach a new programmer one thing it would be this: Program to an interface, not an implementation."; "A good developer codes against abstractions, and not implementations."). Da ist es ein Vorteil, wenn man kein zertifizierter Delphi-Priester ist: Man kommt nicht in den Olymp, kann aber auch nicht auf dem Scheiterhaufen landen. :wink:
|
AW: TObjectList-Object da, aber beim Zugriff Stack-Overflow
@Benmik
Mach dir da nichts draus. Der Nick versteckt die implementierenden Klassen auch gerne mal im
Delphi-Quellcode:
Teil und erzeugt Instanzen im
implementation
Delphi-Quellcode:
Abschnitt.
initialization
Da kenne ich wiederum einige, die ihm dafür die Büßer-Kutte umhängen würden. :stupid: (Ja, ich auch) |
Alle Zeitangaben in WEZ +1. Es ist jetzt 11:09 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