mirror of
https://gitlab.com/freepascal.org/fpc/pas2js.git
synced 2025-04-05 11:17:45 +02:00
* TObjectDictionary
This commit is contained in:
parent
da43764f69
commit
572897dec0
@ -255,7 +255,7 @@ type
|
||||
TMyType = TDictionary<TKey,TValue>;
|
||||
TMyPair = TPair<TKey,TValue>;
|
||||
|
||||
constructor Create(ACapacity: Integer); overload;
|
||||
constructor Create(ACapacity: Integer=0); overload;
|
||||
constructor Create2(const Collection: TEnumerable<TMyPair>); overload;
|
||||
destructor Destroy; override;
|
||||
|
||||
@ -368,6 +368,23 @@ type
|
||||
property OnValueNotify: TCollectionNotifyEvent<TValue> read FOnValueNotify write FOnValueNotify;
|
||||
end;
|
||||
|
||||
TDictionaryOwnership = (doOwnsKeys, doOwnsValues);
|
||||
TDictionaryOwnerships = set of TDictionaryOwnership;
|
||||
|
||||
{ TObjectDictionary }
|
||||
|
||||
TObjectDictionary<TKey,TValue> = class(TDictionary<TKey,TValue>)
|
||||
private
|
||||
FOwnerships: TDictionaryOwnerships;
|
||||
protected
|
||||
Function CanClearMap : Boolean; override;
|
||||
procedure KeyNotify(const Key: TKey; Action: TCollectionNotification); override;
|
||||
procedure ValueNotify(const Value: TValue; Action: TCollectionNotification); override;
|
||||
public
|
||||
constructor Create2(aOwnerships: TDictionaryOwnerships; ACapacity: Integer); overload;
|
||||
constructor Create(aOwnerships: TDictionaryOwnerships); overload;
|
||||
Property OwnerShips : TDictionaryOwnerships Read FOwnerships Write FOwnerShips;
|
||||
end;
|
||||
|
||||
implementation
|
||||
|
||||
@ -1522,4 +1539,44 @@ begin
|
||||
Writeln('Unlocking already unlocked list, lockcount : ',FLock);
|
||||
end;
|
||||
|
||||
{ TObjectDictionary }
|
||||
|
||||
function TObjectDictionary<TKey, TValue>.CanClearMap: Boolean;
|
||||
begin
|
||||
Result:=(Inherited CanClearMap) and (FOwnerships=[]);
|
||||
end;
|
||||
|
||||
procedure TObjectDictionary<TKey, TValue>.KeyNotify(const Key: TKey; Action: TCollectionNotification);
|
||||
|
||||
Var
|
||||
A : TObject absolute key; // Avoid typecast, refused by compiler
|
||||
|
||||
begin
|
||||
inherited KeyNotify(Key, Action);
|
||||
if (doOwnsKeys in FOwnerships) and (Action = cnRemoved) then
|
||||
A.Free;
|
||||
end;
|
||||
|
||||
procedure TObjectDictionary<TKey, TValue>.ValueNotify(const Value: TValue; Action: TCollectionNotification);
|
||||
|
||||
Var
|
||||
A : TObject absolute Value; // Avoid typecast, refused by compiler
|
||||
|
||||
begin
|
||||
inherited ValueNotify(Value, Action);
|
||||
if (doOwnsValues in FOwnerships) and (Action = cnRemoved) then
|
||||
A.Free;
|
||||
end;
|
||||
|
||||
constructor TObjectDictionary<TKey, TValue>.Create2(aOwnerships: TDictionaryOwnerships; ACapacity: Integer);
|
||||
begin
|
||||
Create(aOwnerShips);
|
||||
end;
|
||||
|
||||
constructor TObjectDictionary<TKey, TValue>.Create(aOwnerships: TDictionaryOwnerships);
|
||||
begin
|
||||
Inherited Create;
|
||||
FOwnerShips:=aOwnerships;
|
||||
end;
|
||||
|
||||
end.
|
||||
|
@ -61,8 +61,199 @@ Type
|
||||
procedure TestKeyValueNotificationSet;
|
||||
end;
|
||||
|
||||
{ TMyObject }
|
||||
|
||||
TMyObject = Class(TObject)
|
||||
Private
|
||||
fOnDestroy : TNotifyEvent;
|
||||
FID : Integer;
|
||||
public
|
||||
Constructor Create(aID : Integer; aOnDestroy : TNotifyEvent);
|
||||
destructor destroy; override;
|
||||
Property ID : Integer Read FID;
|
||||
end;
|
||||
|
||||
TSingleObjectDict = Class(Specialize TObjectDictionary<Integer,TMyObject>);
|
||||
TDualObjectDict = Class(Specialize TObjectDictionary<TMyObject,TMyObject>);
|
||||
|
||||
{ TTestSingleObjectDict }
|
||||
|
||||
TTestSingleObjectDict = Class(TTestCase)
|
||||
private
|
||||
FDict: TSingleObjectDict;
|
||||
FList : TFPList;
|
||||
procedure DoAdd(aID: Integer);
|
||||
procedure DoDestroy(Sender: TObject);
|
||||
Public
|
||||
Procedure SetUp; override;
|
||||
Procedure TearDown; override;
|
||||
Property Dict : TSingleObjectDict Read FDict;
|
||||
Published
|
||||
Procedure TestEmpty;
|
||||
Procedure TestFreeOnRemove;
|
||||
Procedure TestNoFreeOnRemove;
|
||||
end;
|
||||
|
||||
TTestDualObjectDict = Class(TTestCase)
|
||||
private
|
||||
FDict: TDualObjectDict;
|
||||
FList : TFPList;
|
||||
procedure DoAdd(aID: Integer);
|
||||
procedure DoDestroy(Sender: TObject);
|
||||
Public
|
||||
Procedure SetUp; override;
|
||||
Procedure TearDown; override;
|
||||
Property Dict : TDualObjectDict Read FDict;
|
||||
Published
|
||||
Procedure TestEmpty;
|
||||
Procedure TestFreeOnRemove;
|
||||
Procedure TestNoFreeOnRemove;
|
||||
end;
|
||||
|
||||
implementation
|
||||
|
||||
{ TTestSingleObjectDict }
|
||||
|
||||
procedure TTestSingleObjectDict.SetUp;
|
||||
begin
|
||||
FDict:=TSingleObjectDict.Create([doOwnsValues]);
|
||||
FList:=TFPList.Create;
|
||||
inherited SetUp;
|
||||
end;
|
||||
|
||||
procedure TTestSingleObjectDict.TearDown;
|
||||
begin
|
||||
FreeAndNil(FDict);
|
||||
FreeAndNil(FList);
|
||||
inherited TearDown;
|
||||
end;
|
||||
|
||||
procedure TTestSingleObjectDict.TestEmpty;
|
||||
begin
|
||||
AssertNotNull('Have object',Dict);
|
||||
AssertEquals('Have empty object',0,Dict.Count);
|
||||
end;
|
||||
|
||||
procedure TTestSingleObjectDict.DoAdd(aID : Integer);
|
||||
|
||||
Var
|
||||
O : TMyObject;
|
||||
|
||||
begin
|
||||
O:=TMyObject.Create(aID,@DoDestroy);
|
||||
FList.Add(O);
|
||||
FDict.Add(aID,O);
|
||||
end;
|
||||
|
||||
procedure TTestSingleObjectDict.DoDestroy(Sender: TObject);
|
||||
|
||||
Var
|
||||
I : Integer;
|
||||
|
||||
begin
|
||||
I:=FList.IndexOf(Sender);
|
||||
AssertTrue('Have object in list',I<>-1);
|
||||
FList.Delete(I);
|
||||
end;
|
||||
|
||||
procedure TTestSingleObjectDict.TestFreeOnRemove;
|
||||
|
||||
begin
|
||||
DoAdd(1);
|
||||
AssertEquals('Have obj',1,FList.Count);
|
||||
Dict.Remove(1);
|
||||
AssertEquals('Have no obj',0,FList.Count);
|
||||
end;
|
||||
|
||||
procedure TTestSingleObjectDict.TestNoFreeOnRemove;
|
||||
begin
|
||||
Dict.OwnerShips:=[];
|
||||
DoAdd(1);
|
||||
AssertEquals('Have obj',1,FList.Count);
|
||||
Dict.Remove(1);
|
||||
AssertEquals('Have obj',1,FList.Count);
|
||||
end;
|
||||
|
||||
{ TTestDualObjectDict }
|
||||
|
||||
procedure TTestDualObjectDict.SetUp;
|
||||
begin
|
||||
FDict:=TDualObjectDict.Create([doOwnsKeys,doOwnsValues]);
|
||||
FList:=TFPList.Create;
|
||||
inherited SetUp;
|
||||
end;
|
||||
|
||||
procedure TTestDualObjectDict.TearDown;
|
||||
begin
|
||||
FreeAndNil(FDict);
|
||||
FreeAndNil(FList);
|
||||
inherited TearDown;
|
||||
end;
|
||||
|
||||
procedure TTestDualObjectDict.TestEmpty;
|
||||
begin
|
||||
AssertNotNull('Have object',Dict);
|
||||
AssertEquals('Have empty object',0,Dict.Count);
|
||||
end;
|
||||
|
||||
procedure TTestDualObjectDict.DoAdd(aID : Integer);
|
||||
|
||||
Var
|
||||
O1,O10 : TMyObject;
|
||||
|
||||
begin
|
||||
O1:=TMyObject.Create(aID,@DoDestroy);
|
||||
FList.Add(O1);
|
||||
O10:=TMyObject.Create(aID*10,@DoDestroy);
|
||||
FList.Add(O10);
|
||||
FDict.Add(O1,O10);
|
||||
end;
|
||||
|
||||
procedure TTestDualObjectDict.DoDestroy(Sender: TObject);
|
||||
|
||||
Var
|
||||
I : Integer;
|
||||
|
||||
begin
|
||||
I:=FList.IndexOf(Sender);
|
||||
AssertTrue('Have object in list',I<>-1);
|
||||
FList.Delete(I);
|
||||
end;
|
||||
|
||||
procedure TTestDualObjectDict.TestFreeOnRemove;
|
||||
|
||||
begin
|
||||
DoAdd(1);
|
||||
AssertEquals('Have obj',2,FList.Count);
|
||||
Dict.Remove(TMyObject(FList[0]));
|
||||
AssertEquals('Have no obj',0,FList.Count);
|
||||
end;
|
||||
|
||||
procedure TTestDualObjectDict.TestNoFreeOnRemove;
|
||||
begin
|
||||
Dict.OwnerShips:=[doOwnsValues];
|
||||
DoAdd(1);
|
||||
AssertEquals('Have obj',2,FList.Count);
|
||||
Dict.Remove(TMyObject(FList[0]));
|
||||
AssertEquals('Have obj',1,FList.Count);
|
||||
AssertEquals('Have key',1,TMyObject(Flist[0]).ID);
|
||||
end;
|
||||
|
||||
{ TMyObject }
|
||||
|
||||
constructor TMyObject.Create(aID: Integer; aOnDestroy: TNotifyEvent);
|
||||
begin
|
||||
FOnDestroy:=aOnDestroy;
|
||||
FID:=AID;
|
||||
end;
|
||||
|
||||
destructor TMyObject.destroy;
|
||||
begin
|
||||
if Assigned(FOnDestroy) then
|
||||
FOnDestroy(Self);
|
||||
inherited destroy;
|
||||
end;
|
||||
|
||||
{ TTestSimpleDictionary }
|
||||
|
||||
procedure TTestSimpleDictionary.SetUp;
|
||||
@ -458,6 +649,8 @@ begin
|
||||
end;
|
||||
|
||||
begin
|
||||
RegisterTest(TTestSimpleDictionary);
|
||||
RegisterTests([TTestSimpleDictionary,
|
||||
TTestSingleObjectDict,
|
||||
TTestDualObjectDict]);
|
||||
end.
|
||||
|
||||
|
@ -80,6 +80,10 @@
|
||||
<Filename Value="tcgenericdictionary.pp"/>
|
||||
<IsPartOfProject Value="True"/>
|
||||
</Unit>
|
||||
<Unit>
|
||||
<Filename Value="../packages/rtl/webutils.pas"/>
|
||||
<IsPartOfProject Value="True"/>
|
||||
</Unit>
|
||||
</Units>
|
||||
</ProjectOptions>
|
||||
<CompilerOptions>
|
||||
|
@ -30,7 +30,7 @@ uses
|
||||
// tcgenarrayhelper,
|
||||
// tcstringhelp
|
||||
tcgenericdictionary,
|
||||
strutils, sysutils;
|
||||
strutils, sysutils, webutils;
|
||||
|
||||
var
|
||||
Application : TTestRunner;
|
||||
@ -41,5 +41,6 @@ begin
|
||||
Application.RunFormClass:=TConsoleRunner;
|
||||
Application.Initialize;
|
||||
Application.Run;
|
||||
|
||||
// Application.Free;
|
||||
end.
|
||||
|
Loading…
Reference in New Issue
Block a user