![]() |
Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Hallo zusammen,
ich habe eine
Delphi-Quellcode:
.
TMyObjectList = class(TObjectList<TMyObject>);
Jedes TMyObject hat eine eindeutige ID (ID aus der Datenbank) Jetzt könnte es sein, dass ich an anderer Stelle bei einer Änderung eine Instanz1.2 bekomme, die schon in der Liste enthalten ist (Instanz1.1). Da es sein kann, dass irgendwo im Code schon auf Instanz1.1 verwiesen wird (in meinem Fall in zwei TVirtualStringTree), kann ich diese ja nicht einfach in der Liste löschen und dafür Instanz1.2 hinzufügen. Wie schaffe ich es aber einfach, dass Instanz1.1 die eventuell aktuelleren Daten aus Instanz1.2 übertragen bekommt? Grüße Gerd |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Mit .Assign() kann man Werte von einem Objekt auf das andere übertragen.
Delphi-Quellcode:
<Instanz1.1>.Assign( <Instanz1.2>);
|
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Aber nur wenn Assign/AssignTo implementiert ist.
Aber das Prinzip stimmt schonmal ... du mußt den Inhalt von 2 in die 1 kopieren / ihr zuweisen ... und sei es, dass du es mit jedem Feld/Property einzeln machen mußt. |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Zitat:
OK, in dem gerade aufgetretenen Fall sind es nur 5 Felder, die sich ändern können und damit ist die Implementation von Assign überschaubar. Es ging mir aber ums generelle |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Um was für Objekte hanldet es sich? Wenn es eigene Klassen sind einfach Assign für diese implementieren.
|
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Meine Klasse sieht prinzipiell so aus. Hier schon mit dem Assign, wie ich es verstanden hätte
Delphi-Quellcode:
Oder geht das einfacher?TMyBaseClass = class; private FID:Integer; FName: String; procedure SetID(const Value: Integer); procedure SetName(const Value: String); public property ID: Integer read FID write SetID; property Name: String read FName write SetName; constructor Create; destructor Destroy; override; procedure Assign(aNewValue:TMyBaseClass); function FunctionB1: String; procedure ProcB1(sMsg: String); end; TMyClass = class(TMyBaseClass); private FInfo:String; procedure SetInfo(const Value: String); public property Info: String read FInfo write SetInfo; constructor Create; destructor Destroy; override; procedure Assign(aNewValue:TMyClass); function Function10: String; procedure Proc10(sMsg: String); end; .. Procedure TMyBaseClass.Assign(aNewValue:TMyBaseClass); begin self.ID := aNewValue.ID; // Und damit Möglichkeit, TMyClass komplett zu überschreiben self.Name := aNewValue.Name; end; Procedure TMyClass.Assign(aNewValue:TMyClass); begin inherited (TMyBaseClass(aNewValue)); self.Info := aNewValue.Info; end; Mein Problem ist, dass TMyClass aus ca. 40 Properties besteht und noch Unter- und Überklassen bestehen. Nur Arbeit, alle Properties in Assign aufzunehemen, aber schwierig, den aktuelle zu halten wenn mehrere eventuell neue Properties hinzufügen (Selten, aber kann vorkommen) |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Anders geht es nunmal nicht (effektiv).
Man kann ab TPersistent vererben und dort das Streaming missbrauchen oder via RTTI alles suchen und kopieren. Bei sowas, wo man etwas mal vergessen könnte, kann man auch
Delphi-Quellcode:
{$IF SizeOf(TMyRecord) <> 123} {$MESSAGE Error '...'} {$IFEND}
Bei Klassen ist das leider nicht ganz so einfach, direkt im Compiler, daher wohl eher
Delphi-Quellcode:
if Self.InstanceSize <> 123 then raise ...;
[edit] Kannst auch mal nachsehn, wo sich die InstanceSize in der TypeInfo versteckt. Das in dein Assign und 123 auf den aktuellen Wert setzen. -> neues Feld = Peng |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
War mir nicht sicher ob Markus mit
Zitat:
|
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Nein.
![]() Ob man nun extern jedes Feld/Property kopiert, oder in einer Methode ala Assign macht ... gemacht werden muß es so oder so. Sie können es auch garnicht, da sie nicht wissen was und wie es kopiert werden muß, sobald es keine einfachen Typen sind. |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Wenn das in allen Klassen implemnetiert ist, musst Du ja nur für die "neuen" Eigenschaften implementieren
Delphi-Quellcode:
TBasis = class
a,b: Integer; procedure Assign( Source: TPersistent); virtual; end; TTeil = class aa, zz: integer; procedure Assign( Source: TPersistent); virtual; end; TAbg = class( TBasis) c: TTeil; d, e: Integer; procedure Assign( Source: TPersistent); virtual; end; ... procedure TBasis.Assign( Source: TPersistent); begin a := Source.a; b := Source.b; end; ... procedure TTeil.Assign( Source: TPersistent); begin aa := Source.aa; zz := Source.zz; end; ... procedure TAbg.Assign( Source: TPersistent); begin inherited Assign( Source); //a, b c.Assign( Source.c); //c d := Source.d; e := Source.e; end; |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Danke für eure Tips. Ich habe bestätigt bekommen, was ich so zu wissen geglaubt habe, aber trotzdem nochmal absichern wollte.
Bei vielen Properties wird leider aus dem "nur nur für die "..neuen" Eigenschaften..." etwas mehr und es geht eher um die laufenden Pflege. Dass jeder auch daran denkt dass da noch ein Assign gepflegt werden muss. Ein klein bisschen hatte ich die Hoffung, dass es irgendwo noch eine Möglichkeit gibt, den Zeiger auf Instanz1.1 so zu ändern, dass er auf Instanz1.2 weiter leitet. |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Delphi-Quellcode:
:angle:
TAbg = class(TBasis)
procedure Assign(Source: TPersistent); {virtual;} overload; Zitat:
aber nja. oder ein Objekt im Object, wo sich das innere Objekt tauschen lässt. |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Zitat:
Nur unsauber oder falsch? |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Das Delphi-eigene Assign ist in TPersistent deklariert und bildet somit schon mal ein Grundgerüst. Dort sind Assign und AssignTo virtual deklariert und können dann in abgeleiteten Klassen überschrieben und mit Leben gefüllt werden. Da deine Klassen nicht von TPersistent abgeleitet sind, hast du da natürlich freie Wahl.
|
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Dort war nur "falsch", dass es in TBasis virtual war und im Nachfahren nochmal virtual,
also verdeckt, anstatt überschrieben. |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Das zweimalige Virtual hatte ich nach deinem Hinweis gesehen. Nur bei der generelle Verwendung war ich mir unsicher.
|
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
ich habe mir jetzt nicht alle Vorschläge durchgelesen...aber wären Interfaces nicht dafür geeignet? Das Objekt wird erstellt, einer Interface-Variablen zugeordnet, in der anderen Liste einer anderen Interface-Variablen zugeordnet usw...bei Änderung des Objekts ist es dann in allen Instanzen sichtbar.
|
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Ich habe (wenn ich das Problem richtig verstehe) in einem ähnlichen Fall einen Pool eingerichtet, der meine Objektinstanzen verwaltet - jede Instanz einmalig.
In buisinessbezogenen Listen werden dann nur Referenzen auf diese Objekte verwaltet (statt TObjectList dann entsprechend TList verwenden). Drei Zugriffsmöglichkeiten sehe ich: - direkter Zugriff auf Objktinstanzen - Zugriff über Interfaces - Zugriff über Guids (der Pool muss dann eine binäre Suche unterstützen) |
AW: Objekt in Liste mit anderer Instanz ersetzen ohne Referenz zu verlieren
Hallo zusammen,
für mich ist der Thread gelöst. Umstellung auf Instanzen oder Pooling sind aktuell keine Lösung. Die Anwendung ist zu umfangreich und damit der Eingriff zu groß. Mein Problem wäre schon damit gelöst, wenn der Teil, der die Änderung vornimmt, auch auf die Liste zugreifen würde. Dann hätte ich gar keine zwei Instanzen. Aber auch das bedeutet Zeit die ich momentan nicht habe. Da war die Synchronisation der notwendigen Felder per Assign schneller implementiert Danke an alle Gerd |
Alle Zeitangaben in WEZ +1. Es ist jetzt 19:41 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