From 8bf49305316acf3b85efda9a234065a5e32bafc7 Mon Sep 17 00:00:00 2001 From: michael Date: Sun, 31 May 2020 09:21:27 +0000 Subject: [PATCH] * Dictionary start --- packages/rtl/generics.collections.pas | 527 +++++++++++++++++++++++++- test/tcgenarrayhelper.pas | 2 +- test/tcgenericdictionary.pp | 202 ++++++++++ test/testrtl.lpi | 9 +- test/testrtl.lpr | 6 +- 5 files changed, 741 insertions(+), 5 deletions(-) create mode 100644 test/tcgenericdictionary.pp diff --git a/packages/rtl/generics.collections.pas b/packages/rtl/generics.collections.pas index bb6c95d..9024f46 100644 --- a/packages/rtl/generics.collections.pas +++ b/packages/rtl/generics.collections.pas @@ -6,7 +6,7 @@ unit Generics.Collections; interface uses - Classes, SysUtils, + Classes, SysUtils, rtlconsts, Types, {$IFDEF Pas2js}JS,{$ENDIF} Generics.Strings, Generics.Defaults; @@ -198,6 +198,151 @@ type property Items[Index: SizeInt]: T read GetItem write SetItem; default; end; + { TPair } + + TPair = record + Key: TKey; + Value: TValue; + constructor Create(const AKey: TKey; const AValue: TValue); + end; + + // Hash table using linear probing + + { TDictionary } + EDictionary = Class(Exception); + + TDictionary = class(TEnumerable>) + private + FMap: TJSMap; + function GetItem(const Key: TKey): TValue; + procedure SetItem(const Key: TKey; const Value: TValue); + procedure DoAdd(const Key: TKey; const Value: TValue); + function DoRemove(const Key: TKey; Notification: TCollectionNotification): TValue; + Function GetCount : Integer; + protected + function DoGetEnumerator: TEnumerator>; override; + procedure PairNotify(const Key: TKey; Value : TValue; Action: TCollectionNotification); virtual; + procedure KeyNotify(const Key: TKey; Action: TCollectionNotification); virtual; + procedure ValueNotify(const Value: TValue; Action: TCollectionNotification); virtual; + public + Type + TMyType = TDictionary; + TMyPair = TPair; + + constructor Create(ACapacity: Integer); overload; + constructor Create2(const Collection: TEnumerable); overload; + destructor Destroy; override; + + procedure Add(const Key: TKey; const Value: TValue); + procedure Remove(const Key: TKey); + function ExtractPair(const Key: TKey): TMyPair; + procedure Clear; + function TryGetValue(const Key: TKey; out Value: TValue): Boolean; + procedure AddOrSetValue(const Key: TKey; const Value: TValue); + function ContainsKey(const Key: TKey): Boolean; + function ContainsValue(const Value: TValue): Boolean; + function ToArray: TArray; override; + + property Items[const Key: TKey]: TValue read GetItem write SetItem; default; + property Count: Integer read GetCount; + + type + { TPairEnumerator } + + TPairEnumerator = class(TEnumerator) + private + FIter: TJSIterator; + FVal : TJSIteratorValue; + function GetCurrent: TMyPair; + protected + function DoGetCurrent: TMyPair; override; + function DoMoveNext: Boolean; override; + public + constructor Create(const ADictionary: TMyType); + function MoveNext: Boolean; reintroduce; + property Current: TMyPair read GetCurrent; + end; + + { TKeyEnumerator } + + TKeyEnumerator = class(TEnumerator) + private + FIter: TJSIterator; + FVal : TJSIteratorValue; + function GetCurrent: TKey; + protected + function DoGetCurrent: TKey; override; + function DoMoveNext: Boolean; override; + public + constructor Create(const AIter: TJSIterator); overload; + constructor Create(const ADictionary: TMyType); overload; + function MoveNext: Boolean; reintroduce; + property Current: TKey read GetCurrent; + end; + + { TValueEnumerator } + + TValueEnumerator = class(TEnumerator) + private + FIter: TJSIterator; + FVal : TJSIteratorValue; + function GetCurrent: TValue; + protected + function DoGetCurrent: TValue; override; + function DoMoveNext: Boolean; override; + public + constructor Create(const AIter: TJSIterator); overload; + constructor Create(const ADictionary: TMyType); overload; + function MoveNext: Boolean; reintroduce; + property Current: TValue read GetCurrent; + end; + + { TValueCollection } + + TValueCollection = class(TEnumerable) + private + FMap: TJSMap; + function GetCount: Integer; + protected + function DoGetEnumerator: TEnumerator; override; + public + constructor Create(const ADictionary: TMyType); + function GetEnumerator: TValueEnumerator; reintroduce; + function ToArray: TArray; override; + property Count: Integer read GetCount; + end; + + { TKeyCollection } + + TKeyCollection = class(TEnumerable) + private + FMap: TJSMap; + function GetCount: Integer; + protected + function DoGetEnumerator: TEnumerator; override; + public + constructor Create(const ADictionary: TMyType); + function GetEnumerator: TKeyEnumerator; reintroduce; + function ToArray: TArray; override; + property Count: Integer read GetCount; + end; + + private + FOnKeyNotify: TCollectionNotifyEvent; + FOnValueNotify: TCollectionNotifyEvent; + FKeyCollection: TKeyCollection; + FValueCollection: TValueCollection; + function GetKeys: TKeyCollection; + function GetValues: TValueCollection; + public + function GetEnumerator: TPairEnumerator; reintroduce; + property Keys: TKeyCollection read GetKeys; + property Values: TValueCollection read GetValues; + property OnKeyNotify: TCollectionNotifyEvent read FOnKeyNotify write FOnKeyNotify; + property OnValueNotify: TCollectionNotifyEvent read FOnValueNotify write FOnValueNotify; + end; + + implementation { TCustomArrayHelper } @@ -864,4 +1009,384 @@ begin Result := TMyArrayHelper.BinarySearch(FItems, AItem, AIndex, AComparer, 0, Count); end; +{ TPair } + +constructor TPair.Create(const AKey: TKey; const AValue: TValue); +begin + Key:=aKey; + Value:=aValue; +end; + +{ TDictionary } + +ResourceString + SErrDictKeyNotFound = 'Key value not found'; + SErrDictDuplicateKey = 'Duplicate key value'; + +function TDictionary.GetItem(const Key: TKey): TValue; + +Var + V : JSValue; + +begin + V:=FMap.Get(Key); + if isUndefined(v) then + Raise EDictionary.Create(SErrDictKeyNotFound); + Result:=TValue(V); +end; + +procedure TDictionary.SetItem(const Key: TKey; const Value: TValue); + +Var + V : JSValue; + +begin + V:=FMap.Get(Key); + if isUndefined(v) then + ValueNotify(TValue(V),cnRemoved); + FMap.&Set(Key,Value); + ValueNotify(Value, cnAdded); +end; + +procedure TDictionary.DoAdd(const Key: TKey; const Value: TValue); +begin + FMap.&Set(Key,Value); + KeyNotify(Key,cnAdded); + ValueNotify(Value,cnAdded); +end; + + +function TDictionary.DoRemove(const Key: TKey; Notification: TCollectionNotification): TValue; +Var + V : JSValue; + +begin + V:=FMap.Get(Key); + if Not isUndefined(v) then + begin + FMap.Delete(Key); + Result:=TValue(v); + KeyNotify(Key,Notification); + ValueNotify(Result,Notification); + end; +end; + +function TDictionary.GetCount: Integer; +begin + Result:=FMap.Size; +end; + +function TDictionary.DoGetEnumerator: TEnumerator; +begin + Result:=TPairEnumerator.Create(Self); +end; + +procedure TDictionary.PairNotify(const Key: TKey; Value : TValue; Action: TCollectionNotification); + +begin + KeyNotify(Key,action); + ValueNotify(Value,action); +end; + +procedure TDictionary.KeyNotify(const Key: TKey; Action: TCollectionNotification); +begin + if Assigned(FOnKeyNotify) then + FOnKeyNotify(Self,Key,Action); +end; + +procedure TDictionary.ValueNotify(const Value: TValue; Action: TCollectionNotification); +begin + if Assigned(FOnValueNotify) then + FOnValueNotify(Self,Value,Action); +end; + +constructor TDictionary.Create(ACapacity: Integer = 0); +begin + + FMap:=TJSMap.New; +end; + +constructor TDictionary.Create2(const Collection: TEnumerable); + +Var + aPair : TMyPair; + +begin + Create(0); + For aPair in Collection do + Add(aPair.Key,aPair.Value); +end; + +destructor TDictionary.Destroy; +begin + FreeAndNil(FKeyCollection); + FreeAndNil(FValueCollection); + Clear; + FMap:=Nil; + inherited Destroy; +end; + +procedure TDictionary.Add(const Key: TKey; const Value: TValue); +begin + if FMap.Has(Key) then + Raise EDictionary.Create(SErrDictDuplicateKey); + DoAdd(Key,Value); +end; + +procedure TDictionary.Remove(const Key: TKey); +begin + doRemove(Key,cnRemoved); +end; + +function TDictionary.ExtractPair(const Key: TKey): TMyPair; + +begin + if FMap.Has(Key) then + begin + Result.Create(Key,TValue(FMap.get(key))); + FMap.Delete(Key); + end + else + Result.Create(Key,Default(TValue)); +end; + +procedure TDictionary.Clear; +begin + FMap.Clear; +end; + + +function TDictionary.TryGetValue(const Key: TKey; out Value: TValue): Boolean; +begin + Result:=FMap.Has(Key); + If Result then + Value:=TValue(FMap.get(Key)); +end; + +procedure TDictionary.AddOrSetValue(const Key: TKey; const Value: TValue); +begin + if Not FMap.Has(Key) then + DoAdd(Key,Value) + else + SetItem(Key,Value); +end; + +function TDictionary.ContainsKey(const Key: TKey): Boolean; +begin + Result:=FMap.Has(Key); +end; + +function TDictionary.ContainsValue(const Value: TValue): Boolean; + +Var + It : TJSIterator; + Res : TJSIteratorValue; + +begin + Result:=False; + It:=FMap.Values; + Repeat + Res:=It.next; + if not Res.done then + Result:=(Value=TValue(Res.value)); + Until (Result or Res.done); +end; + +function TDictionary.ToArray: TArray; +begin + Result:=inherited ToArray; +end; + +function TDictionary.GetKeys: TKeyCollection; +begin + if FKeyCollection=Nil then + FKeyCollection:=TKeyCollection.Create(Self); + Result:=FKeyCollection; +end; + +function TDictionary.GetValues: TValueCollection; +begin + if FValueCollection=Nil then + FValueCollection:=TValueCollection.Create(Self); + Result:=FValueCollection; +end; + +function TDictionary.GetEnumerator: TPairEnumerator; +begin + Result:=TPairEnumerator.Create(Self); +end; + +{ TDictionary.TPairEnumerator } + +function TDictionary.TPairEnumerator.GetCurrent: TMyPair; +begin + Result:=DoGetCurrent; +end; + +function TDictionary.TPairEnumerator.DoGetCurrent: TMyPair; + +Var + A : TJSValueDynArray; + +begin + A:=TJSValueDynArray(FVal.Value); + Result.Create(TKey(A[0]),TValue(A[1])); +end; + +function TDictionary.TPairEnumerator.DoMoveNext: Boolean; +begin + FIter.Next; + Result:=Not FVal.Done; +end; + +constructor TDictionary.TPairEnumerator.Create(const ADictionary: TMyType); +begin + FIter:=ADictionary.FMap.Entries; +end; + +function TDictionary.TPairEnumerator.MoveNext: Boolean; +begin + Result:=DoMoveNext; +end; + +{ TDictionary.TKeyEnumerator } + +function TDictionary.TKeyEnumerator.GetCurrent: TKey; +begin + Result:=DoGetCurrent; +end; + +function TDictionary.TKeyEnumerator.DoGetCurrent: TKey; +begin + Result:=TKey(FVal.Value); +end; + +function TDictionary.TKeyEnumerator.DoMoveNext: Boolean; +begin + FIter.Next; + Result:=Not FVal.Done; +end; + +constructor TDictionary.TKeyEnumerator.Create(const ADictionary: TMyType); +begin + Create(ADictionary.FMap.Keys); +end; + +constructor TDictionary.TKeyEnumerator.Create(const AIter : TJSIterator); +begin + FIter:=aIter; +end; + +function TDictionary.TKeyEnumerator.MoveNext: Boolean; +begin + Result:=DoMoveNext; +end; + +{ TDictionary.TValueEnumerator } + +function TDictionary.TValueEnumerator.GetCurrent: TValue; +begin + Result:=DoGetCurrent; +end; + +function TDictionary.TValueEnumerator.DoGetCurrent: TValue; +begin + Result:=TValue(FVal.Value); +end; + +function TDictionary.TValueEnumerator.DoMoveNext: Boolean; +begin + FIter.Next; + Result:=Not FVal.Done; +end; + +constructor TDictionary.TValueEnumerator.Create(const ADictionary: TMyType); +begin + Create(aDictionary.FMap.Values); +end; + +constructor TDictionary.TValueEnumerator.Create(const AIter: TJSIterator); +begin + FIter:=AIter; +end; + +function TDictionary.TValueEnumerator.MoveNext: Boolean; +begin + Result:=DoMoveNext; +end; + +{ TDictionary.TValueCollection } + +function TDictionary.TValueCollection.GetCount: Integer; +begin + Result:=FMap.Size; +end; + +function TDictionary.TValueCollection.DoGetEnumerator: TEnumerator; +begin + Result:=TValueEnumerator.Create(FMap.Values); +end; + +constructor TDictionary.TValueCollection.Create(const ADictionary: TMyType); +begin + FMap:=ADictionary.FMap; +end; + +function TDictionary.TValueCollection.GetEnumerator: TValueEnumerator; +begin + Result:=TValueEnumerator(DoGetEnumerator); +end; + +function TDictionary.TValueCollection.ToArray: TArray; + +Var + I : Integer; + P : TValue; + +begin + SetLength(Result,FMap.Size); + For P in Self do + begin + Result[i]:=P; + Inc(I); + End; +end; + +{ TDictionary.TKeyCollection } + +function TDictionary.TKeyCollection.GetCount: Integer; +begin + Result:=FMap.Size; +end; + +function TDictionary.TKeyCollection.DoGetEnumerator: TEnumerator; +begin + Result:=GetEnumerator; +end; + +constructor TDictionary.TKeyCollection.Create(const ADictionary: TMyType); +begin + FMap:=aDictionary.FMap; +end; + +function TDictionary.TKeyCollection.GetEnumerator: TKeyEnumerator; +begin + Result:=TKeyEnumerator.Create(FMap.Keys); +end; + +function TDictionary.TKeyCollection.ToArray: TArray; +begin + Result:=inherited ToArray; +end; + +Type + TMyDict = TDictionary; + +Var + MyDict : TMyDict; + +begin + MyDict:=TMyDict.Create; + MyDict.Add(1,'aloga'); + MyDict.Free; end. diff --git a/test/tcgenarrayhelper.pas b/test/tcgenarrayhelper.pas index 460d16d..364b955 100644 --- a/test/tcgenarrayhelper.pas +++ b/test/tcgenarrayhelper.pas @@ -58,7 +58,7 @@ procedure TTestArrayHelper.CheckBinarySearch(constref AArray: TArray; begin CheckEquals(AExpectedResult, - TArrayHelper.BinarySearch(AArray,AValue,ASearchResult,aComparer), + TArrayHelper.BinarySearch(AArray,AValue,ASearchResult,aComparer,0,Length(AArray)), 'Wrong BinarySearch result for ' + AValue.ToString); end; diff --git a/test/tcgenericdictionary.pp b/test/tcgenericdictionary.pp new file mode 100644 index 0000000..4855a8c --- /dev/null +++ b/test/tcgenericdictionary.pp @@ -0,0 +1,202 @@ +unit tcgenericdictionary; + +{$mode objfpc} + +interface + +uses + fpcunit, testregistry, Classes, SysUtils, Generics.Defaults, Generics.Collections; + +Type + TMySimpleDict = Class(Specialize TDictionary); + + { TTestSimpleDictionary } + + TTestSimpleDictionary = Class(TTestCase) + Private + FDict : TMySimpleDict; + procedure DoAdd(aCount: Integer; aOffset: Integer=0); + procedure DoAdd2; + procedure DoGetValue(aKey: Integer; Match: String; ExceptionClass: TClass=nil); + Public + Procedure SetUp; override; + Procedure TearDown; override; + Property Dict : TMySimpleDict Read FDict; + Published + Procedure TestEmpty; + Procedure TestAdd; + Procedure TestClear; + Procedure TestTryGetValue; + Procedure TestGetValue; + Procedure TestSetValue; + Procedure TestAddDuplicate; + Procedure TestAddOrSet; + Procedure TestContainsKey; + Procedure TestDelete; + end; + +implementation + +{ TTestSimpleDictionary } + +procedure TTestSimpleDictionary.SetUp; +begin + inherited SetUp; + FDict:=TMySimpleDict.Create; +end; + +procedure TTestSimpleDictionary.TearDown; +begin + FreeAndNil(FDict); + inherited TearDown; +end; + +procedure TTestSimpleDictionary.TestEmpty; +begin + AssertNotNull('Have dictionary',Dict); + AssertEquals('empty dictionary',0,Dict.Count); +end; + +procedure TTestSimpleDictionary.DoAdd(aCount : Integer; aOffset : Integer=0); + +Var + I : Integer; + +begin + if aOffset=-1 then + aOffset:=Dict.Count; + For I:=aOffset+1 to aOffset+aCount do + Dict.Add(I,IntToStr(i)); +end; + +procedure TTestSimpleDictionary.TestAdd; + +begin + DoAdd(1); + AssertEquals('Count OK',1,Dict.Count); + AssertTrue('Has added value',Dict.ContainsKey(1)); + DoAdd(1,1); + AssertEquals('Count OK',2,Dict.Count); + AssertTrue('Has added value',Dict.ContainsKey(2)); +end; + +procedure TTestSimpleDictionary.TestClear; +begin + DoAdd(3); + AssertEquals('Count OK',3,Dict.Count); + Dict.Clear; + AssertEquals('Count after clear OK',0,Dict.Count); +end; + +procedure TTestSimpleDictionary.TestTryGetValue; + +Var + I : integer; + SI,A : string; + +begin + DoAdd(3); + For I:=1 to 3 do + begin + SI:=IntToStr(I); + AssertTrue('Have value '+SI,Dict.TryGetValue(I,A)); + AssertEquals('Value is correct '+SI,SI,A); + end; + AssertFalse('Have no value 4',Dict.TryGetValue(4,A)); +end; + +procedure TTestSimpleDictionary.DoGetValue(aKey: Integer; Match: String; ExceptionClass: TClass); + +Var + EC : TClass; + A,EM : String; + +begin + EC:=Nil; + try + A:=Dict.Items[aKey]; + except + On E : Exception do + begin + EC:=E.ClassType; + EM:=E.Message; + end + end; + if ExceptionClass=Nil then + begin + if EC<>Nil then + Fail('Got exception '+EC.ClassName+' with message: '+EM); + AssertEquals('Value is correct for '+IntToStr(aKey),Match,A) + end + else + begin + if EC=Nil then + Fail('Expected exception '+ExceptionClass.ClassName+' but got none'); + if EC<>ExceptionClass then + Fail('Expected exception class '+ExceptionClass.ClassName+' but got '+EC.ClassName+' with message '+EM); + end; +end; + +procedure TTestSimpleDictionary.TestGetValue; + +Var + I : integer; + +begin + DoAdd(3); + For I:=1 to 3 do + DoGetValue(I,IntToStr(I)); + DoGetValue(4,'4',EDictionary); +end; + +procedure TTestSimpleDictionary.TestSetValue; +begin + TestGetValue; + Dict.Items[3]:='Six'; + DoGetValue(3,'Six'); +end; + +procedure TTestSimpleDictionary.DoAdd2; + +begin + Dict.Add(2,'A new 2'); +end; + +procedure TTestSimpleDictionary.TestAddDuplicate; +begin + DoAdd(3); + AssertException('Cannot add duplicate',EDictionary,@DoAdd2); +end; + +procedure TTestSimpleDictionary.TestAddOrSet; + +begin + DoAdd(3); + Dict.AddOrSetValue(2,'a new 2'); + DoGetValue(2,'a new 2'); +end; + +procedure TTestSimpleDictionary.TestContainsKey; + +Var + I : Integer; + +begin + DoAdd(3); + For I:=1 to 3 do + AssertTrue('Has '+IntToStr(i),Dict.ContainsKey(I)); + AssertFalse('Has not 4',Dict.ContainsKey(4)); +end; + +procedure TTestSimpleDictionary.TestDelete; +begin + DoAdd(3); + Dict.Remove(2); + AssertEquals('Count',2,Dict.Count); + AssertFalse('Has not 2',Dict.ContainsKey(2)); +end; + +begin + RegisterTest(TTestSimpleDictionary); +end. + diff --git a/test/testrtl.lpi b/test/testrtl.lpi index 4d54885..0ab8313 100644 --- a/test/testrtl.lpi +++ b/test/testrtl.lpi @@ -76,6 +76,10 @@ + + + + @@ -95,6 +99,9 @@ + + + @@ -103,7 +110,7 @@ - + diff --git a/test/testrtl.lpr b/test/testrtl.lpr index 92434b3..c85cd67 100644 --- a/test/testrtl.lpr +++ b/test/testrtl.lpr @@ -27,14 +27,16 @@ program testrtl; uses browserconsole, consoletestrunner, frmrtlrun, // tcstream, tccompstreaming, simplelinkedlist, tcsyshelpers - tcgenarrayhelper, +// tcgenarrayhelper, // tcstringhelp - strutils; + tcgenericdictionary, + strutils, sysutils; var Application : TTestRunner; begin + SysUtils.HookUncaughtExceptions; Application:=TTestRunner.Create(nil); Application.RunFormClass:=TConsoleRunner; Application.Initialize;