| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921 | {    This file is part of the Free Pascal/NewPascal run time library.    Copyright (c) 2018 by Maciej Izak (hnb),    member of the NewPascal development team (http://newpascal.org)    Copyright(c) 2004-2018 DaThoX    It contains tests for the Free Pascal generics library    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.    Acknowledgment    Thanks to Sphere 10 Software (http://sphere10.com) for sponsoring    many new types, tests and major refactoring of entire library    Thanks to Castle Game Engine (https://castle-engine.sourceforge.io)    Part of tests for this module was copied from Castle Game Engine tests **********************************************************************}unit tests.generics.stdcollections;{$mode delphi}interfaceuses  fpcunit, testutils, testregistry, tests.generics.utils,  Classes, SysUtils, Generics.Collections, Generics.Defaults;type  TTestStdCollections = class(TTestCollections)  private    procedure Test_TList_Notification(AList: TList<string>); overload;  published    // Tests from Castle Game Engine    procedure Test_List;    procedure Test_FreeingManually;    procedure Test_AddingLists;    procedure Test_Sort;    procedure Test_Pack;    procedure Test_RecordsList;    procedure Test_VectorsList;    procedure Test_MethodsList;    // My (c) tests    procedure Test_SortedList;    procedure Test_Queue;    procedure Test_GenericListBox;    procedure Test_TList_Notification; overload;    procedure Test_TSortedList_Notification;    procedure Test_TQueue_Notification;    procedure Test_TStack_Notification;    procedure Test_TObjectList_Notification;    procedure Test_TObjectQueue_Notification;    procedure Test_TObjectStack_Notification;    procedure Test_TrimExcess;  end;  TGenericListBox<T> = class  private class var    F : TList<TComponentClass>;    class procedure Test(ATest: TTestCase);  end;implementationclass procedure TGenericListBox<T>.Test(ATest: TTestCase);begin  F := TList<TComponentClass>.Create;  F.Add(TDataModule);  F.Add(nil);  with TList<TComponentClass>.Create(F) do  begin    ATest.AssertTrue(Count = 2);    ATest.AssertTrue(F[0] = Items[0]);    ATest.AssertTrue(F[1] = Items[1]);    ATest.AssertTrue(F[0] = TDataModule);    ATest.AssertTrue(F[1] = nil);    Free;  end;  F.Free;end;type  TApple = class    Name: string;  end;type  TAppleList = class(TObjectList<TApple>)    procedure Pack;  end;procedure TAppleList.Pack;begin  while Remove(nil) <> -1 do ;end;procedure TTestStdCollections.Test_List;var  A: TApple;  Apples: TAppleList;begin  Apples := TAppleList.Create(true);  try    A := TApple.Create;    Apples.Add(A);    Apples.Add(TApple.Create);    A := TApple.Create;    Apples.Add(A);    AssertEquals(3, Apples.Count);    AssertEquals(2, Apples.IndexOf(A));    Apples.Delete(0);    AssertEquals(2, Apples.Count);    AssertEquals(1, Apples.IndexOf(A));    Apples.Remove(A);    AssertEquals(1, Apples.Count);    Apples.Delete(0);    AssertEquals(0, Apples.Count);  finally FreeAndNil(Apples) end;end;procedure TTestStdCollections.Test_FreeingManually;var  A: TApple;  Apples: TAppleList;begin  Apples := TAppleList.Create(false);  try    A := TApple.Create;    Apples.Add(A);    Apples.Add(A);    Apples.Add(TApple.Create);    { This freeing would be invalid on a list that owns children,      as we free something twice, and we leave some invalid references      (to already freed items) in the list at various stages.      But it should be OK with list that has OwnsChildren = false. }    Apples[0].Free;    Apples[0] := nil;    Apples[1] := nil;    Apples[2].Free;  finally FreeAndNil(Apples) end;end;procedure TTestStdCollections.Test_AddingLists;var  A: TApple;  Apples, Apples2: TAppleList;begin  Apples := TAppleList.Create(true);  try    A := TApple.Create;    A.Name := 'One';    Apples.Add(A);    A := TApple.Create;    A.Name := 'Two';    Apples.Add(A);    Apples2 := TAppleList.Create(false);    try      Apples2.AddRange(Apples);      Apples2.AddRange(Apples);      Apples2.AddRange(Apples);      AssertEquals(6, Apples2.Count);      AssertEquals('One', Apples2[0].Name);      AssertEquals('Two', Apples2[1].Name);      AssertEquals('One', Apples2[2].Name);      AssertEquals('Two', Apples2[3].Name);      AssertEquals('One', Apples2[4].Name);      AssertEquals('Two', Apples2[5].Name);    finally FreeAndNil(Apples2) end;  finally FreeAndNil(Apples) end;end;function CompareApples(constref Left, Right: TApple): Integer;begin  Result := AnsiCompareStr(Left.Name, Right.Name);end;procedure TTestStdCollections.Test_Sort;type  TAppleComparer = TComparer<TApple>;var  A: TApple;  L: TAppleList;begin  L := TAppleList.Create(true);  try    A := TApple.Create;    A.Name := '11';    L.Add(A);    A := TApple.Create;    A.Name := '33';    L.Add(A);    A := TApple.Create;    A.Name := '22';    L.Add(A);    L.Sort(TAppleComparer.Construct(@CompareApples));    AssertEquals(3, L.Count);    AssertEquals('11', L[0].Name);    AssertEquals('22', L[1].Name);    AssertEquals('33', L[2].Name);  finally FreeAndNil(L) end;end;procedure TTestStdCollections.Test_Pack;var  A: TApple;  L: TAppleList;begin  L := TAppleList.Create(true);  try    L.Add(nil);    A := TApple.Create;    A.Name := '11';    L.Add(A);    L.Add(nil);    A := TApple.Create;    A.Name := '33';    L.Add(A);    A := TApple.Create;    A.Name := '22';    L.Add(A);    L.Add(nil);    L.Add(nil);    L.Pack;    AssertEquals(3, L.Count);    AssertEquals('11', L[0].Name);    AssertEquals('33', L[1].Name);    AssertEquals('22', L[2].Name);  finally FreeAndNil(L) end;end;procedure TTestStdCollections.Test_RecordsList;type  TMyRecord = packed record    A, B: Integer;  end;  TMyRecordList = TList<TMyRecord>;var  List: TMyRecordList;  R1, R2, R: TMyRecord;begin  List := TMyRecordList.Create;  try    R1.A := 11;    R1.B := 22;    List.Add(R1);    R2.A := 33;    R2.B := 44;    List.Add(R2);    R2.A := 33;    R2.B := 44;    List.Add(R2);    AssertEquals(3, List.Count);    AssertEquals(11, List[0].A);    AssertEquals(22, List[0].B);    AssertEquals(33, List[1].A);    AssertEquals(44, List[1].B);    AssertEquals(33, List[2].A);    AssertEquals(44, List[2].B);    List.Delete(2);    AssertEquals(2, List.Count);    AssertEquals(11, List[0].A);    AssertEquals(22, List[0].B);    AssertEquals(33, List[1].A);    AssertEquals(44, List[1].B);    AssertEquals(0, List.IndexOf(R1));    AssertEquals(1, List.IndexOf(R2));    // change R1 and R2, to make sure it doesn't matter for tests    R1.A := 111111;    R1.B := 222222;    R2.A := 333333;    R2.B := 444444;    AssertEquals(-1, List.IndexOf(R1));    AssertEquals(-1, List.IndexOf(R2));    R.A := 11;    R.B := 22;    AssertEquals(0, List.IndexOf(R));    R.A := 33;    R.B := 44;    AssertEquals(1, List.IndexOf(R));    R.A := 11;    R.B := 22;    List.Remove(R);    AssertEquals(1, List.Count);    AssertEquals(33, List[0].A);    AssertEquals(44, List[0].B);    R.A := 666;    R.B := 22;    List.Remove(R); // does nothing, no such record    AssertEquals(1, List.Count);    AssertEquals(33, List[0].A);    AssertEquals(44, List[0].B);  finally FreeAndNil(List) end;end;procedure TTestStdCollections.Test_VectorsList;type  TMyVector = packed array [0..1] of Single;  TMyVectorList = TList<TMyVector>;var  List: TMyVectorList;  R1, R2, R: TMyVector;begin  List := TMyVectorList.Create;  try    R1[0] := 11;    R1[1] := 22;    List.Add(R1);    R2[0] := 33;    R2[1] := 44;    List.Add(R2);    R2[0] := 33;    R2[1] := 44;    List.Add(R2);    AssertEquals(3, List.Count);    AssertEquals(11, List[0][0]);    AssertEquals(22, List[0][1]);    AssertEquals(33, List[1][0]);    AssertEquals(44, List[1][1]);    AssertEquals(33, List[2][0]);    AssertEquals(44, List[2][1]);    List.Delete(2);    AssertEquals(2, List.Count);    AssertEquals(11, List[0][0]);    AssertEquals(22, List[0][1]);    AssertEquals(33, List[1][0]);    AssertEquals(44, List[1][1]);    AssertEquals(0, List.IndexOf(R1));    AssertEquals(1, List.IndexOf(R2));    // change R1 and R2, to make sure it doesn't matter for tests    R1[0] := 111111;    R1[1] := 222222;    R2[0] := 333333;    R2[1] := 444444;    AssertEquals(-1, List.IndexOf(R1));    AssertEquals(-1, List.IndexOf(R2));    R[0] := 11;    R[1] := 22;    AssertEquals(0, List.IndexOf(R));    R[0] := 33;    R[1] := 44;    AssertEquals(1, List.IndexOf(R));    R[0] := 11;    R[1] := 22;    List.Remove(R);    AssertEquals(1, List.Count);    AssertEquals(33, List[0][0]);    AssertEquals(44, List[0][1]);    R[0] := 666;    R[1] := 22;    List.Remove(R); // does nothing, no such item    AssertEquals(1, List.Count);    AssertEquals(33, List[0][0]);    AssertEquals(44, List[0][1]);  finally FreeAndNil(List) end;end;type  TSomeClass = class    procedure Foo(A: Integer);  end;procedure TSomeClass.Foo(A: Integer);beginend;procedure TTestStdCollections.Test_MethodsList;type  TMyMethod = procedure (A: Integer) of object;  TMyMethodList = TList<TMyMethod>;  procedure AssertMethodsEqual(const M1, M2: TMyMethod);  begin    AssertTrue(TMethod(M1).Code = TMethod(M2).Code);    AssertTrue(TMethod(M1).Data = TMethod(M2).Data);  end;var  List: TMyMethodList;  C1, C2, C3: TSomeClass;  M: TMyMethod;begin  C1 := TSomeClass.Create;  C2 := TSomeClass.Create;  C3 := TSomeClass.Create;  List := TMyMethodList.Create;  try    List.Add(C1.Foo);    List.Add(C2.Foo);    List.Add(C2.Foo);    AssertEquals(3, List.Count);    M := C1.Foo;    AssertMethodsEqual(List[0], M);    M := C2.Foo;    AssertMethodsEqual(List[1], M);    AssertMethodsEqual(List[2], M);    List.Delete(2);    AssertEquals(2, List.Count);    M := C1.Foo;    AssertMethodsEqual(List[0], M);    M := C2.Foo;    AssertMethodsEqual(List[1], M);    AssertEquals(0, List.IndexOf(C1.Foo));    AssertEquals(1, List.IndexOf(C2.Foo));    AssertEquals(-1, List.IndexOf(C3.Foo));    List.Remove(C1.Foo);    AssertEquals(1, List.Count);    M := C2.Foo;    AssertMethodsEqual(List[0], M);    List.Remove(C3.Foo); // does nothing, no such item    AssertEquals(1, List.Count);    M := C2.Foo;    AssertMethodsEqual(List[0], M);  finally FreeAndNil(List) end;  C1.Free;  C2.Free;  C3.Free;end;procedure TTestStdCollections.Test_SortedList;var  LSortedList: TSortedList<Integer>;  i: integer;  LRandomOrder: TArray<Integer>;begin  LRandomOrder := TArray<Integer>.Create(    10, 8, 17, 19, 2, 0, 13, 15, 5, 7, 12, 14, 4, 6, 11, 9, 16, 18, 3, 1);  LSortedList := TSortedList<Integer>.Create;  for i in LRandomOrder do    LSortedList.Add(i);  AssertEquals('Wrong Count value for TSortedList', Length(LRandomOrder), LSortedList.Count);  for i := 0 to 19 do    AssertEquals(Format('Wrong item (%d) index (%d) in TSortedList',[LSortedList[i], i]), i, LSortedList[i]);  LSortedList.Free;end;procedure TTestStdCollections.Test_Queue;const  NUMBERS: array[0..2] of Integer = (3,4,5);var  LQueue: TQueue<Integer>;  i: Integer;  j: Integer;  pi: Pinteger;begin  LQueue := TQueue<Integer>.Create;  for i := 1 to 5 do  begin    LQueue.Enqueue(i);    AssertEquals(LQueue.Peek, 1);  end;  AssertEquals(LQueue.Dequeue, 1);  AssertEquals(LQueue.Extract, 2);  j := 0;  for i in LQueue do  begin    AssertEquals(i, NUMBERS[j]);    Inc(j);  end;  j := 0;  for pi in LQueue.Ptr^ do  begin    AssertEquals(pi^, NUMBERS[j]);    Inc(j);  end;  LQueue.Free;end;procedure TTestStdCollections.Test_TList_Notification(AList: TList<string>);var  LStringsObj: TEnumerable<string>;  LStringsIntf: IEnumerable<string>;begin  try    LStringsObj := EnumerableStringsObj(['Ddd', 'Eee']);    LStringsIntf := EnumerableStringsIntf(['Fff', 'Ggg']);    AList.OnNotify := NotifyTestStr;    { Add + AddRange }    NotificationAdd(AList, ['Aaa', 'Bbb', 'Ccc', 'Ddd', 'Eee', 'Fff', 'Ggg'], cnAdded);    AList.Add('Aaa');    AList.AddRange(['Bbb', 'Ccc']);    AList.AddRange(LStringsObj);    AList.AddRange(LStringsIntf);    AssertNotificationsExecutedStr;    { Clear }    NotificationAdd(AList, ['Aaa', 'Bbb', 'Ccc', 'Ddd', 'Eee', 'Fff', 'Ggg'], cnRemoved);    AList.Clear;    AssertNotificationsExecutedStr;    { Insert + InsertRange }    NotificationAdd(AList, ['Aaa', 'Bbb', 'Ccc', 'Ddd', 'Eee', 'Fff', 'Ggg'], cnAdded);    AList.Insert(0, 'Aaa');    AList.InsertRange(1, ['Bbb', 'Ccc']);    AList.InsertRange(3, LStringsObj);    AList.InsertRange(5, LStringsIntf);    AssertNotificationsExecutedStr;    { Remove + Delete + DeleteRange }    NotificationAdd(AList, ['Aaa', 'Bbb', 'Ccc', 'Ddd', 'Eee', 'Fff', 'Ggg'], cnRemoved);    AList.Remove('Aaa');    AList.Delete(0);    AList.DeleteRange(0, 5);    AssertEquals(AList.Count, 0);    AssertNotificationsExecutedStr;    { ExtractIndex, Extract }    NotificationAdd(AList, ['Aaa', 'Bbb', 'Ccc'], cnAdded);    AList.AddRange(['Aaa', 'Bbb', 'Ccc']);    AssertNotificationsExecutedStr;    NotificationAdd(AList, ['Aaa', 'Bbb'], cnExtracted);    AssertEquals(AList.ExtractIndex(0), 'Aaa');    AssertEquals(AList.Extract('Bbb'), 'Bbb');    AssertNotificationsExecutedStr;    { SetItem }    NotificationAdd(AList, 'Ccc', cnRemoved);    NotificationAdd(AList, 'FPC', cnAdded);    AList[0] := 'FPC';    AssertNotificationsExecutedStr;  finally    LStringsObj.Free;    { Free }    NotificationAdd(AList, 'FPC', cnRemoved);    AList.Free;    AssertNotificationsExecutedStr;  end;end;procedure TTestStdCollections.Test_TList_Notification;begin  Test_TList_Notification(TList<string>.Create);end;procedure TTestStdCollections.Test_TSortedList_Notification;var  LList: TSortedList<string>;begin  LList := TSortedList<string>.Create;  LList.SortStyle := cssUser;  Test_TList_Notification(LList);end;procedure TTestStdCollections.Test_TQueue_Notification;var  LQueue: TQueue<string>;begin  LQueue := TQueue<string>.Create();  try    LQueue.OnNotify := NotifyTestStr;    { Enqueue }    NotificationAdd(LQueue, ['Aaa', 'Bbb', 'Ccc', 'Ddd'], cnAdded);    LQueue.Enqueue('Aaa');    LQueue.Enqueue('Bbb');    LQueue.Enqueue('Ccc');    LQueue.Enqueue('Ddd');    AssertNotificationsExecutedStr;    { Dequeue }    NotificationAdd(LQueue, 'Aaa', cnRemoved);    AssertEquals(LQueue.Dequeue, 'Aaa');    AssertNotificationsExecutedStr;    { Extract }    NotificationAdd(LQueue, 'Bbb', cnExtracted);    AssertEquals(LQueue.Extract, 'Bbb');    AssertNotificationsExecutedStr;    { Clear }    NotificationAdd(LQueue, ['Ccc', 'Ddd'], cnRemoved);    LQueue.Clear;    AssertNotificationsExecutedStr;    { Enqueue }    NotificationAdd(LQueue, ['FPC', 'Polandball'], cnAdded);    LQueue.Enqueue('FPC');    LQueue.Enqueue('Polandball');    AssertNotificationsExecutedStr;  finally    NotificationAdd(LQueue, ['FPC', 'Polandball'], cnRemoved);    LQueue.Free;    AssertNotificationsExecutedStr;  end;end;procedure TTestStdCollections.Test_TStack_Notification;var  LStack: TStack<string>;begin  LStack := TStack<string>.Create();  try    LStack.OnNotify := NotifyTestStr;    { Push }    NotificationAdd(LStack, ['Aaa', 'Bbb', 'Ccc', 'Ddd'], cnAdded);    LStack.Push('Aaa');    LStack.Push('Bbb');    LStack.Push('Ccc');    LStack.Push('Ddd');    AssertNotificationsExecutedStr;    { Pop }    NotificationAdd(LStack, 'Ddd', cnRemoved);    AssertEquals(LStack.Pop, 'Ddd');    AssertNotificationsExecutedStr;    { Extract }    NotificationAdd(LStack, 'Ccc', cnExtracted);    AssertEquals(LStack.Extract, 'Ccc');    AssertNotificationsExecutedStr;    { Clear }    NotificationAdd(LStack, ['Bbb', 'Aaa'], cnRemoved);    LStack.Clear;    AssertNotificationsExecutedStr;    { Push }    NotificationAdd(LStack, ['FPC', 'Polandball'], cnAdded);    LStack.Push('FPC');    LStack.Push('Polandball');    AssertNotificationsExecutedStr;  finally    NotificationAdd(LStack, ['Polandball', 'FPC'], cnRemoved);    LStack.Free;    AssertNotificationsExecutedStr;  end;end;procedure TTestStdCollections.Test_TObjectList_Notification;var  LObj: TEnumerable<TObject>;  LIntf: IEnumerable<TObject>;  O: TArray<TObject>;  LList: TObjectList<TObject>;  i: Integer;begin  try    CreateObjects(O, 8);    LList := TObjectList<TObject>.Create(false);    LList.OnNotify := NotifyTestObj;    LObj := EnumerableObjectsObj([O[3], O[4]]);    LIntf := EnumerableObjectsIntf([O[5], O[6]]);    { Add + AddRange }    NotificationAdd(LList, [O[0], O[1], O[2], O[3], O[4], O[5], O[6]], cnAdded);    LList.Add(O[0]);    LList.AddRange([O[1], O[2]]);    LList.AddRange(LObj);    LList.AddRange(LIntf);    AssertNotificationsExecutedObj;    { Clear }    NotificationAdd(LList, [O[0], O[1], O[2], O[3], O[4], O[5], O[6]], cnRemoved);    LList.Clear;    AssertNotificationsExecutedObj;    { Insert + InsertRange }    NotificationAdd(LList, [O[0], O[1], O[2], O[3], O[4], O[5], O[6]], cnAdded);    LList.Insert(0, O[0]);    LList.InsertRange(1, [O[1], O[2]]);    LList.InsertRange(3, LObj);    LList.InsertRange(5, LIntf);    AssertNotificationsExecutedObj;    { Remove + Delete + DeleteRange }    NotificationAdd(LList, [O[0], O[1], O[2], O[3], O[4], O[5], O[6]], cnRemoved);    LList.Remove(O[0]);    LList.Delete(0);    LList.DeleteRange(0, 5);    AssertEquals(LList.Count, 0);    AssertNotificationsExecutedObj;    { ExtractIndex, Extract }    NotificationAdd(LList, [O[0], O[1], O[2]], cnAdded);    LList.AddRange([O[0], O[1], O[2]]);    AssertNotificationsExecutedObj;    NotificationAdd(LList, [O[0], O[1]], cnExtracted);    AssertTrue(LList.ExtractIndex(0) = O[0]);    AssertTrue(LList.Extract(O[1]) = O[1]);    AssertNotificationsExecutedObj;    { SetItem }    NotificationAdd(LList, O[2], cnRemoved);    NotificationAdd(LList, O[7], cnAdded);    LList[0] := O[7];    AssertNotificationsExecutedObj;  finally    LObj.Free;    { Free }    NotificationAdd(LList, O[7], cnRemoved);    FreeObjects(O);    LList.Free;    AssertNotificationsExecutedObj;  end;end;procedure TTestStdCollections.Test_TObjectQueue_Notification;var  LQueue: TObjectQueue<TObject>;  O: TArray<TObject>;begin  LQueue := TObjectQueue<TObject>.Create(false);  try    CreateObjects(O, 6);    LQueue.OnNotify := NotifyTestObj;    { Enqueue }    NotificationAdd(LQueue, [O[0], O[1], O[2], O[3]], cnAdded);    LQueue.Enqueue(O[0]);    LQueue.Enqueue(O[1]);    LQueue.Enqueue(O[2]);    LQueue.Enqueue(O[3]);    AssertNotificationsExecutedObj;    { Dequeue }    NotificationAdd(LQueue, O[0], cnRemoved);    LQueue.Dequeue;    AssertNotificationsExecutedObj;    { Extract }    NotificationAdd(LQueue, O[1], cnExtracted);    AssertTrue(LQueue.Extract = O[1]);    AssertNotificationsExecutedObj;    { Clear }    NotificationAdd(LQueue, [O[2], O[3]], cnRemoved);    LQueue.Clear;    AssertNotificationsExecutedObj;    { Enqueue }    NotificationAdd(LQueue, [O[4], O[5]], cnAdded);    LQueue.Enqueue(O[4]);    LQueue.Enqueue(O[5]);    AssertNotificationsExecutedObj;  finally    NotificationAdd(LQueue, [O[4], O[5]], cnRemoved);    FreeObjects(O);    LQueue.Free;    AssertNotificationsExecutedObj;  end;end;procedure TTestStdCollections.Test_TObjectStack_Notification;var  LStack: TStack<TObject>;  O: TArray<TObject>;begin  LStack := TObjectStack<TObject>.Create(false);  try    CreateObjects(O, 6);    LStack.OnNotify := NotifyTestObj;    { Push }    NotificationAdd(LStack, [O[0], O[1], O[2], O[3]], cnAdded);    LStack.Push(O[0]);    LStack.Push(O[1]);    LStack.Push(O[2]);    LStack.Push(O[3]);    AssertNotificationsExecutedObj;    { Pop }    NotificationAdd(LStack, O[3], cnRemoved);    AssertTrue(LStack.Pop = O[3]);    AssertNotificationsExecutedObj;    { Extract }    NotificationAdd(LStack, O[2], cnExtracted);    AssertTrue(LStack.Extract = O[2]);    AssertNotificationsExecutedObj;    { Clear }    NotificationAdd(LStack, [O[1], O[0]], cnRemoved);    LStack.Clear;    AssertNotificationsExecutedObj;    { Pop }    NotificationAdd(LStack, [O[4], O[5]], cnAdded);    LStack.Push(O[4]);    LStack.Push(O[5]);    AssertNotificationsExecutedObj;  finally    NotificationAdd(LStack, [O[5], O[4]], cnRemoved);    FreeObjects(O);    LStack.Free;    AssertNotificationsExecutedObj;  end;end;procedure TTestStdCollections.Test_GenericListBox;begin  TGenericListBox<Integer>.Test(Self);end;procedure TTestStdCollections.Test_TrimExcess;var  LList: TList<Integer>;  LQueue: TQueue<Integer>;  LStack: TStack<Integer>;begin  LList := TList<Integer>.Create;  LQueue := TQueue<Integer>.Create;  LStack := TStack<Integer>.Create;  try    LList.AddRange([1, 2, 3, 4, 5, 6]);    LList.DeleteRange(2, 3);    CheckNotEquals(LList.Capacity, LList.Count);    LList.TrimExcess;    AssertEquals(LList.Capacity, LList.Count);    LQueue.Enqueue(1);    LQueue.Enqueue(2);    LQueue.Dequeue;    CheckNotEquals(LQueue.Capacity, LQueue.Count);    LQueue.TrimExcess;    AssertEquals(LQueue.Capacity, LQueue.Count);    LStack.Push(1);    LStack.Push(2);    LStack.Pop;    CheckNotEquals(LStack.Capacity, LStack.Count);    LStack.TrimExcess;    AssertEquals(LStack.Capacity, LStack.Count);  finally    LStack.Free;    LQueue.Free;    LList.Free;  end;end;begin  RegisterTest(TTestStdCollections);end.
 |