Gestern war ich ja mal richtig optimistisch, dass ich da bald zu einer Lösung kommme, aber der Optimismus schwindet.
Vielleicht liegt es ja an Delphi XE
Das Positive: Mit dem neuen Konstrukt funktioniert es zunächst mal einwandfrei
- Zugriff auf lNamen[i].Namen ist möglich
- FChanged der Liste wird gesetzt sobald ich lNamen[i].Namen ändere
Dann aber die Fehler
1. Sobald ich das Programm Schließe, kommt weiterhin der EAccessViolation. Ohne TBaseObjectList<T>.Destructor gleich beim lNamenListFree, mit Destructor nach dem OnClose
2. Das Abfangen, ob lNamen Mitglied einer Liste, funktioniert nicht. Auch wenn ich FParentProject explizit auf NIL setze, geht er in den Message-Block
Delphi-Quellcode:
constructor TBase.Create;
begin
FParentObject := nil;
end;
procedure TBase.SetChanged(const Value: Boolean);
var
msg: TDispatchMessage;
begin
if FChanged <> Value then
begin
FChanged := Value;
if Value and (FParentObject <> nil) then // Geht in den Block auch wenn FParentObject = NIL
begin
msg.MsgID := cNotifyChanged;
FParentObject.Dispatch(msg);
end;
end;
end;
3. Ganz seltsam wird es dann in folgenden Blöcken
Delphi-Quellcode:
procedure TForm15.btCreateListClick(Sender: TObject);
// *****************************************************************************************************************************************
// Simuliert Erstellung neuer Einträge
var
i: Integer;
lNamen: TNamen;
begin
if (lNamenList = nil) then
lNamenList := TNamenList.Create;
for i := 0 to 5 do
begin
lNamen.Create;
lNamenList.Add(lNamen);
lNamen.ParentObject := lNamenList;
lNamen.Namen := Format('Name%d', [i]);
lNamen.Changed := False;
end;
lNamenList.Changed := False;
laChanged.Caption := Format('lNamenList Count = %d, Changed = %d', [lNamenList.Count, Integer(lNamenList.Changed)]);
end;
procedure TForm15.btNameOnlyClick(Sender: TObject);
// *****************************************************************************************************************************************
// Simuliert StandAlone-Klasse, d.h. nicht in einer Liste
var
lNamen: TNamen;
begin
lNamen.Create;
try
lNamen.Namen := 'NameOnly';
finally
lNamen.Free;
end;
end;
procedure TForm15.btNewNameClick(Sender: TObject);
// *****************************************************************************************************************************************
// Simuliert Hinzufügen eines Namens
var
lNamen: TNamen;
begin
lNamen.Create;
lNamenList.Add(lNamen);
lNamen.ParentObject := lNamenList;
lNamen.Namen := Format('Name%d', [lNamenList.Count + 1]);
laChanged.Caption := Format('lNamenList Count = %d, Changed = %d', [lNamenList.Count, Integer(lNamenList.Changed)]);
end;
- btCreateListClick funktioniert einwandfrei auch bei mehrfachem Aufruf
- btNameOnlyClick und btNewNameClick führen zu einem EAccessViolation in TNamen.Create. Dabei ist btNewNameClick doch genau gleich wie btCreateListClick mit dem einzigen Unterschied dass einmal 1 und einmal 6 neue Daten erstellt und in Liste eingefügt werden