/usr/lib/lazarus/0.9.30.4/lcl/dynhasharray.pp is in lazarus-src-0.9.30.4 0.9.30.4-6.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 | {
Author: Mattias Gaertner
*****************************************************************************
* *
* This file is part of the Lazarus Component Library (LCL) *
* *
* See the file COPYING.modifiedLGPL.txt, 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. *
* *
*****************************************************************************
Abstract:
This unit defines TDynHashArray, which is very similar to a TList, since
it also stores pointer/objects.
It supports Add, Remove, Contains, First, Count and Clear.
Because of the hashing nature the operations adding, removing and finding is
done in constant time on average.
Inner structure:
There are three parts:
1. The array itself (FItems). Every entry is a pointer to the first
TDynHashArrayItem of a list with the same hash index. The first item
of every same index list is the list beginning and its IsOverflow
flag is set to false. All other items are overflow items.
To get all items with the same hash index, do a FindHashItem. Then
search through all "Next" items until Next is nil or its IsOverflow
flag is set to false.
2. The items beginning with FFirstItem is a 2-way-connected list of
TDynHashArrayItem. This list contains all used items.
3. To reduce GetMem/FreeMem calls, free items are cached.
Issues:
The maximum capacity is the PrimeNumber. You can store more items, but the
performance decreases. The best idea is to provide your own hash function.
Important: Items in the TDynHashArray must not change their key.
When changing the key of an item, remove it and add it after the change.
}
unit DynHashArray;
{$Mode ObjFPC}{$H+}
interface
uses Classes, SysUtils, LCLProc;
type
TDynHashArray = class;
THashFunction = function(Sender: TDynHashArray; Item: Pointer): integer;
TOwnerHashFunction = function(Item: Pointer): integer of object;
TOnGetKeyForHashItem = function(Item: pointer): pointer;
TOnEachHashItem = function(Sender: TDynHashArray; Item: Pointer): boolean;
PDynHashArrayItem = ^TDynHashArrayItem;
TDynHashArrayItem = record
Item: Pointer;
Next, Prior: PDynHashArrayItem;
IsOverflow: boolean;
end;
TDynHashArrayOption = (dhaoCachingEnabled, dhaoCacheContains);
TDynHashArrayOptions = set of TDynHashArrayOption;
{ TDynHashArray }
TDynHashArray = class
private
FItems: ^PDynHashArrayItem;
FCount: integer;
FCapacity: integer;
FMinCapacity: integer;
FMaxCapacity: integer;
FFirstItem: PDynHashArrayItem;
FHashCacheItem: Pointer;
FHashCacheIndex: integer;
FLowWaterMark: integer;
FHighWaterMark: integer;
FCustomHashFunction: THashFunction;
FOnGetKeyForHashItem: TOnGetKeyForHashItem;
FOptions: TDynHashArrayOptions;
FOwnerHashFunction: TOwnerHashFunction;
FContainsCache: TObject;
function NewHashItem: PDynHashArrayItem;
procedure DisposeHashItem(ADynHashArrayItem: PDynHashArrayItem);
procedure ComputeWaterMarks;
procedure SetCapacity(NewCapacity: integer);
procedure SetCustomHashFunction(const AValue: THashFunction);
procedure SetOnGetKeyForHashItem(const AValue: TOnGetKeyForHashItem);
procedure SetOptions(const AValue: TDynHashArrayOptions);
procedure SetOwnerHashFunction(const AValue: TOwnerHashFunction);
protected
procedure RebuildItems;
procedure SaveCacheItem(Item: Pointer; Index: integer);
public
constructor Create;
constructor Create(InitialMinCapacity: integer);
destructor Destroy; override;
procedure Add(Item: Pointer);
function Contains(Item: Pointer): boolean;
function ContainsKey(Key: Pointer): boolean;
procedure Remove(Item: Pointer);
procedure Clear;
procedure ClearCache;
function First: Pointer;
property Count: integer read fCount;
function IndexOf(AnItem: Pointer): integer;
function IndexOfKey(Key: Pointer): integer;
function FindHashItem(Item: Pointer): PDynHashArrayItem;
function FindHashItemWithKey(Key: Pointer): PDynHashArrayItem;
function FindItemWithKey(Key: Pointer): Pointer;
function GetHashItem(HashIndex: integer): PDynHashArrayItem;
procedure Delete(ADynHashArrayItem: PDynHashArrayItem);
procedure AssignTo(List: TList);
procedure AssignTo(List: TFPList);
procedure ForEach(const Func: TOnEachHashItem);
function SlowAlternativeHashMethod(Sender: TDynHashArray;
Item: Pointer): integer;
function ConsistencyCheck: integer;
procedure WriteDebugReport;
property FirstHashItem: PDynHashArrayItem read FFirstItem;
property MinCapacity: integer read FMinCapacity write FMinCapacity;
property MaxCapacity: integer read FMaxCapacity write FMaxCapacity;
property Capacity: integer read FCapacity;
property CustomHashFunction: THashFunction
read FCustomHashFunction write SetCustomHashFunction;
property OwnerHashFunction: TOwnerHashFunction
read FOwnerHashFunction write SetOwnerHashFunction;
property OnGetKeyForHashItem: TOnGetKeyForHashItem
read FOnGetKeyForHashItem write SetOnGetKeyForHashItem;
property Options: TDynHashArrayOptions read FOptions write SetOptions;
end;
TDynHashArrayItemMemManager = class
private
FFirstFree: PDynHashArrayItem;
FFreeCount: integer;
FCount: integer;
FMinFree: integer;
FMaxFreeRatio: integer;
procedure SetMaxFreeRatio(NewValue: integer);
procedure SetMinFree(NewValue: integer);
procedure DisposeFirstFreeItem;
public
procedure DisposeItem(ADynHashArrayItem: PDynHashArrayItem);
function NewItem: PDynHashArrayItem;
property MinimumFreeCount: integer read FMinFree write SetMinFree;
property MaximumFreeRatio: integer
read FMaxFreeRatio write SetMaxFreeRatio; // in one eighth steps
property Count: integer read FCount;
procedure Clear;
constructor Create;
destructor Destroy; override;
function ConsistencyCheck: integer;
procedure WriteDebugReport;
end;
EDynHashArrayException = class(Exception);
const
ItemMemManager: TDynHashArrayItemMemManager = nil;
implementation
function GetItemMemManager: TDynHashArrayItemMemManager;
begin
if ItemMemManager=nil then
ItemMemManager:=TDynHashArrayItemMemManager.Create;
Result:=ItemMemManager;
end;
const
PrimeNumber: integer = 5364329;
type
TRecentList = class
private
FCapacity: integer;
FCount: integer;
FItems: PPointer;
procedure FreeItems;
procedure SetCapacity(NewCapacity: integer);
public
constructor Create(TheCapacity: integer);
destructor Destroy; override;
function Contains(Item: Pointer): boolean;
procedure Add(Item: Pointer);
procedure Remove(Item: Pointer);
function IndexOf(Item: Pointer): integer;
procedure Clear;
function ConsistencyCheck: integer;
property Cacpacity: integer read FCapacity;
property Count: integer read FCount;
end;
{ TRecentList }
procedure TRecentList.FreeItems;
begin
if FItems<>nil then begin
FreeMem(FItems);
FItems:=nil;
end;
end;
procedure TRecentList.SetCapacity(NewCapacity: integer);
begin
if NewCapacity=FCapacity then exit;
if NewCapacity>0 then
ReAllocMem(FItems,NewCapacity*SizeOf(Pointer))
else
FreeItems;
FCapacity:=NewCapacity;
if FCount>FCapacity then FCount:=FCapacity;
end;
constructor TRecentList.Create(TheCapacity: integer);
begin
inherited Create;
if TheCapacity<1 then FCapacity:=1;
SetCapacity(TheCapacity);
end;
destructor TRecentList.Destroy;
begin
FreeItems;
inherited Destroy;
end;
function TRecentList.Contains(Item: Pointer): boolean;
begin
Result:=IndexOf(Item)>=0;
end;
procedure TRecentList.Add(Item: Pointer);
begin
if FCount=FCapacity then begin
if FCount>1 then
Move(FItems[1],FItems[0],SizeOf(PPointer)*(FCount-1));
end else begin
inc(FCount);
end;
FItems[FCount-1]:=Item;
end;
procedure TRecentList.Remove(Item: Pointer);
var i: integer;
begin
i:=IndexOf(Item);
if i<0 then exit;
if i<FCount-1 then
Move(FItems[i+1],FItems[i],SizeOf(PPointer)*(FCount-i-1));
dec(FCount);
end;
function TRecentList.IndexOf(Item: Pointer): integer;
begin
Result:=FCount-1;
while (Result>=0) and (FItems[Result]<>Item) do dec(Result);
end;
procedure TRecentList.Clear;
begin
FCount:=0;
end;
function TRecentList.ConsistencyCheck: integer;
begin
if FCount>FCapacity then exit(-1);
if FCapacity=0 then exit(-2);
if FItems=nil then exit(-3);
Result:=0;
end;
{ TDynHashArray }
procedure TDynHashArray.WriteDebugReport;
var i, RealHashIndex: integer;
HashItem: PDynHashArrayItem;
begin
DebugLn('TDynHashArray.WriteDebugReport: Consistency=',dbgs(ConsistencyCheck));
DebugLn(' Count=',dbgs(FCount),' Capacity=',dbgs(FCapacity));
for i:=0 to FCapacity-1 do begin
HashItem:=FItems[i];
if HashItem<>nil then begin
DbgOut(' Index=',IntToStr(i));
while HashItem<>nil do begin
DbgOut(' ',Dbgs(HashItem^.Item));
RealHashIndex:=IndexOf(HashItem^.Item);
if RealHashIndex<>i then DbgOut('(H='+dbgs(RealHashIndex)+')');
HashItem:=HashItem^.Next;
if (HashItem<>nil) and (HashItem^.IsOverflow=false) then break;
end;
DebugLn;
end;
end;
HashItem:=FFirstItem;
while HashItem<>nil do begin
DebugLn(' ',Dbgs(HashItem^.Prior),'<-'
,Dbgs(HashItem)
,'(',Dbgs(HashItem^.Item),')'
,'->',Dbgs(HashItem^.Next));
HashItem:=HashItem^.Next;
end;
end;
constructor TDynHashArray.Create(InitialMinCapacity: integer);
var Size: integer;
begin
inherited Create;
FMinCapacity:=InitialMinCapacity;
FMaxCapacity:=PrimeNumber;
if FMinCapacity<5 then FMinCapacity:=137;
FCapacity:=FMinCapacity;
Size:=FCapacity * SizeOf(TDynHashArrayItem);
GetMem(FItems,Size);
FillChar(FItems^,Size,0);
FCount:=0;
FFirstItem:=nil;
ComputeWaterMarks;
FHashCacheIndex:=-1;
end;
destructor TDynHashArray.Destroy;
begin
Clear;
FreeMem(FItems);
FContainsCache.Free;
inherited Destroy;
end;
function TDynHashArray.ConsistencyCheck: integer;
var RealCount, i: integer;
HashItem, HashItem2: PDynHashArrayItem;
OldCacheItem: pointer;
OldCacheIndex: integer;
begin
RealCount:=0;
// check first item
if (FFirstItem<>nil) and (FFirstItem^.IsOverflow) then
exit(-1);
if (FItems=nil) and (FFirstItem<>nil) then
exit(-2);
// check for doubles and circles
HashItem:=FFirstItem;
while HashItem<>nil do begin
HashItem2:=HashItem^.Prior;
while HashItem2<>nil do begin
if HashItem=HashItem2 then
exit(-3); // circle
if HashItem^.Item=HashItem2^.Item then
exit(-4); // double item
HashItem2:=HashItem2^.Prior;
end;
HashItem:=HashItem^.Next;
end;
// check chain
HashItem:=FFirstItem;
while HashItem<>nil do begin
inc(RealCount);
if (HashItem^.Next<>nil) and (HashItem^.Next^.Prior<>HashItem) then
exit(-6);
if (HashItem^.Prior<>nil) and (HashItem^.Prior^.Next<>HashItem) then
exit(-7);
if (HashItem^.IsOverflow=false)
and (FItems[IndexOf(HashItem^.Item)]<>HashItem) then
exit(-8);
HashItem:=HashItem^.Next;
end;
// check count
if RealCount<>FCount then exit(-9);
// check FItems
RealCount:=0;
for i:=0 to FCapacity-1 do begin
HashItem:=FItems[i];
while HashItem<>nil do begin
inc(RealCount);
if IndexOf(HashItem^.Item)<>i then exit(-14);
HashItem:=HashItem^.Next;
if (HashItem<>nil) and (HashItem^.IsOverflow=false) then break;
end;
end;
if RealCount<>FCount then exit(-15);
// check cache
if FHashCacheIndex>=0 then begin
OldCacheItem:=FHashCacheItem;
OldCacheIndex:=FHashCacheIndex;
ClearCache;
FHashCacheIndex:=IndexOfKey(OldCacheItem);
if FHashCacheIndex<>OldCacheIndex then exit(-16);
FHashCacheItem:=OldCacheItem;
end;
// check ContainsCache
if (FContainsCache<>nil) xor (dhaoCacheContains in Options) then exit(-17);
if (FContainsCache<>nil) then begin
Result:=TRecentList(FContainsCache).ConsistencyCheck;
if Result<>0 then begin
dec(Result,100);
exit;
end;
end;
Result:=0;
end;
procedure TDynHashArray.ComputeWaterMarks;
begin
FLowWaterMark:=FCapacity div 4;
FHighWaterMark:=(FCapacity*3) div 4;
end;
function TDynHashArray.IndexOf(AnItem: Pointer): integer;
begin
if (AnItem<>nil) and (FItems<>nil) then begin
if Assigned(OnGetKeyForHashItem) then begin
AnItem:=OnGetKeyForHashItem(AnItem);
end;
Result:=IndexOfKey(AnItem);
end else
Result:=-1;
end;
function TDynHashArray.IndexOfKey(Key: Pointer): integer;
begin
if (FItems<>nil)
and ((Key<>nil) or Assigned(OnGetKeyForHashItem)) then begin
if (dhaoCachingEnabled in Options)
and (Key=FHashCacheItem) and (FHashCacheIndex>=0) then
exit(FHashCacheIndex);
if not Assigned(FCustomHashFunction) then begin
if not Assigned(FOwnerHashFunction) then begin
Result:=Integer((PtrUInt(Key)+(PtrUint(Key) mod 17)) mod Cardinal(FCapacity));
end else
Result:=FOwnerHashFunction(Key);
end else
Result:=FCustomHashFunction(Self,Key);
{if (Key=FHashCacheItem) and (FHashCacheIndex>=0)
and (Result<>FHashCacheIndex) then begin
DebugLn(' DAMN: ',HexStr(PtrInt(Key),8),' ',FHashCacheIndex,'<>',Result);
raise Exception.Create('GROSSER MIST');
end;}
// Check if the owner or custon function has returned something valid
if (Result < 0)
or (Result >= FCapacity)
then raise EDynHashArrayException.CreateFmt('Invalid index %d for key %p', [Result, Key]);
end else
Result:=-1;
end;
procedure TDynHashArray.Clear;
begin
ClearCache;
while FFirstItem<>nil do Delete(FFirstItem);
end;
procedure TDynHashArray.ClearCache;
begin
FHashCacheIndex:=-1;
if FContainsCache<>nil then TRecentList(FContainsCache).Clear;
end;
procedure TDynHashArray.Add(Item: Pointer);
var Index: integer;
HashItem: PDynHashArrayItem;
begin
if Item=nil then exit;
if FCount>=FHighWaterMark then begin
SetCapacity(FCapacity*2-1);
end;
Index:=IndexOf(Item);
if Index < 0 then Exit;
HashItem:=NewHashItem;
HashItem^.Item:=Item;
if FItems[Index]=nil then begin
HashItem^.Next:=FFirstItem;
end else begin
HashItem^.Next:=FItems[Index];
HashItem^.Prior:=HashItem^.Next^.Prior;
HashItem^.Next^.IsOverflow:=true;
end;
if (HashItem^.Next=FFirstItem) then
FFirstItem:=HashItem;
FItems[Index]:=HashItem;
if HashItem^.Next<>nil then begin
HashItem^.Next^.Prior:=HashItem;
if HashItem^.Prior<>nil then
HashItem^.Prior^.Next:=HashItem;
end;
inc(FCount);
SaveCacheItem(Item,Index);
if FContainsCache<>nil then TRecentList(FContainsCache).Clear;
end;
function TDynHashArray.SlowAlternativeHashMethod(Sender: TDynHashArray;
Item: Pointer): integer;
begin
Result:=integer((PtrUInt(Item) mod Cardinal(PrimeNumber))
+(PtrUInt(Item) mod 17)+(PtrUInt(Item) mod 173)
+(PtrUInt(Item) mod 521)
) mod FCapacity;
end;
procedure TDynHashArray.Remove(Item: Pointer);
begin
Delete(FindHashItem(Item));
end;
procedure TDynHashArray.Delete(ADynHashArrayItem: PDynHashArrayItem);
var Index: integer;
OldNext: PDynHashArrayItem;
begin
if ADynHashArrayItem=nil then exit;
// delete from cache
if (FHashCacheIndex>=0)
and ((ADynHashArrayItem^.Item=FHashCacheItem)
or (Assigned(OnGetKeyForHashItem)
and (OnGetKeyForHashItem(ADynHashArrayItem^.Item)=FHashCacheItem)))
then
// if the user removes an item, changes the key and readds it, the hash
// of the item can change
// => the cache must be cleared
ClearCache;
// delete from FItems
if not ADynHashArrayItem^.IsOverflow then begin
// Item is first item with hash
Index:=IndexOf(ADynHashArrayItem^.Item);
if Index < 0 then Exit; // should not happen
OldNext:=ADynHashArrayItem^.Next;
if (OldNext=nil) or (not (OldNext^.IsOverflow)) then
FItems[Index]:=nil
else begin
FItems[Index]:=OldNext;
OldNext^.IsOverflow:=false;
end;
end;
// adjust FFirstItem
if FFirstItem=ADynHashArrayItem then
FFirstItem:=FFirstItem^.Next;
// free storage item
DisposeHashItem(ADynHashArrayItem);
// adjust count and capacity
dec(FCount);
if FCount<FLowWaterMark then begin
// resize
SetCapacity((FCapacity+1) div 2);
end;
end;
procedure TDynHashArray.AssignTo(List: TList);
var
i: integer;
HashItem: PDynHashArrayItem;
begin
List.Count:=Count;
HashItem:=FirstHashItem;
i:=0;
while HashItem<>nil do begin
List[i]:=HashItem^.Item;
inc(i);
HashItem:=HashItem^.Next;
end;
end;
procedure TDynHashArray.AssignTo(List: TFPList);
var
i: integer;
HashItem: PDynHashArrayItem;
begin
List.Count:=Count;
HashItem:=FirstHashItem;
i:=0;
while HashItem<>nil do begin
List[i]:=HashItem^.Item;
inc(i);
HashItem:=HashItem^.Next;
end;
end;
procedure TDynHashArray.ForEach(const Func: TOnEachHashItem);
var
HashItem: PDynHashArrayItem;
begin
HashItem:=FFirstItem;
while HashItem<>nil do begin
if not Func(Self,HashItem^.Item) then break;
HashItem:=HashItem^.Next;
end;
end;
function TDynHashArray.First: Pointer;
begin
if FFirstItem<>nil then
Result:=FFirstItem^.Item
else
Result:=nil;
end;
function TDynHashArray.NewHashItem: PDynHashArrayItem;
begin
Result:=GetItemMemManager.NewItem;
end;
procedure TDynHashArray.DisposeHashItem(ADynHashArrayItem: PDynHashArrayItem);
begin
GetItemMemManager.DisposeItem(ADynHashArrayItem);
end;
function TDynHashArray.Contains(Item: Pointer): boolean;
begin
if (FContainsCache=nil) or (not TRecentList(FContainsCache).Contains(Item))
then begin
Result:=FindHashItem(Item)<>nil;
if Result and (FContainsCache<>nil) then
TRecentList(FContainsCache).Add(Item);
end else
Result:=true;
end;
function TDynHashArray.ContainsKey(Key: Pointer): boolean;
begin
Result:=FindHashItemWithKey(Key)<>nil;
end;
function TDynHashArray.FindHashItem(Item: Pointer): PDynHashArrayItem;
var Index: integer;
begin
if (Item<>nil) and (FItems<>nil) then begin
Index:=IndexOf(Item);
if Index>=0 then begin
Result:=FItems[Index];
if (Result<>nil) then begin
while (Result^.Item<>Item) do begin
Result:=Result^.Next;
if Result=nil then exit;
if Result^.IsOverflow=false then begin
Result:=nil;
exit;
end;
end;
SaveCacheItem(Item,Index);
end;
end else
Result:=nil;
end else
Result:=nil;
end;
function TDynHashArray.FindHashItemWithKey(Key: Pointer): PDynHashArrayItem;
var Index: integer;
begin
if FItems<>nil then begin
Index:=IndexOfKey(Key);
if Index>=0 then begin
Result:=FItems[Index];
if (Result<>nil) then begin
if Assigned(OnGetKeyForHashItem) then begin
if OnGetKeyForHashItem(Result^.Item)=Key then exit;
// search in overflow hash items
Result:=Result^.Next;
while (Result<>nil) and (Result^.IsOverflow) do begin
if OnGetKeyForHashItem(Result^.Item)=Key then begin
FHashCacheIndex:=Index;
FHashCacheItem:=Key;
exit;
end;
Result:=Result^.Next;
end;
Result:=nil;
end;
end;
end else
Result:=nil;
end else
Result:=nil;
end;
function TDynHashArray.FindItemWithKey(Key: Pointer): Pointer;
var
Index: integer;
HashItem: PDynHashArrayItem;
begin
Result:=nil;
if FItems<>nil then begin
Index:=IndexOfKey(Key);
if Index < 0 then Exit; // should not happen
HashItem:=FItems[Index];
if (HashItem<>nil)
and Assigned(OnGetKeyForHashItem) then begin
if OnGetKeyForHashItem(HashItem^.Item)=Key then exit;
// search in overflow hash items
HashItem:=HashItem^.Next;
while (HashItem<>nil) and (HashItem^.IsOverflow) do begin
if OnGetKeyForHashItem(HashItem^.Item)=Key then begin
FHashCacheIndex:=Index;
FHashCacheItem:=Key;
Result:=HashItem^.Item;
exit;
end;
HashItem:=HashItem^.Next;
end;
end;
end;
end;
function TDynHashArray.GetHashItem(HashIndex: integer): PDynHashArrayItem;
begin
Result:=FItems[HashIndex];
end;
procedure TDynHashArray.SetCapacity(NewCapacity: integer);
var Size: integer;
begin
if NewCapacity<FMinCapacity then NewCapacity:=FMinCapacity;
if NewCapacity>FMaxCapacity then NewCapacity:=FMaxCapacity;
if NewCapacity=FCapacity then exit;
// resize FItems
FreeMem(FItems);
FCapacity:=NewCapacity;
Size:=FCapacity * SizeOf(PDynHashArrayItem);
GetMem(FItems,Size);
ComputeWaterMarks;
// rebuild
RebuildItems;
end;
procedure TDynHashArray.SetCustomHashFunction(const AValue: THashFunction);
begin
if FCustomHashFunction=AValue then exit;
FCustomHashFunction:=AValue;
FOwnerHashFunction:=nil;
RebuildItems;
end;
procedure TDynHashArray.SetOwnerHashFunction(const AValue: TOwnerHashFunction);
begin
if FOwnerHashFunction=AValue then exit;
FCustomHashFunction:=nil;
FOwnerHashFunction:=AValue;
RebuildItems;
end;
procedure TDynHashArray.RebuildItems;
var Index: integer;
CurHashItem, NextHashItem: PDynHashArrayItem;
begin
FillChar(FItems^,FCapacity * SizeOf(PDynHashArrayItem),0);
ClearCache;
CurHashItem:=FFirstItem;
FFirstItem:=nil;
while CurHashItem<>nil do begin
NextHashItem:=CurHashItem^.Next;
Index:=IndexOf(CurHashItem^.Item);
if Index < 0
then begin
// ??? something bad happenend
// should we dispose current item ?
// Anyhow, skip it.
CurHashItem := NextHashItem;
Continue;
end;
CurHashItem^.IsOverFlow:=false;
CurHashItem^.Prior:=nil;
if FItems[Index]=nil then begin
CurHashItem^.Next:=FFirstItem;
end else begin
CurHashItem^.Next:=FItems[Index];
CurHashItem^.Prior:=CurHashItem^.Next^.Prior;
CurHashItem^.Next^.IsOverflow:=true;
end;
if (CurHashItem^.Next=FFirstItem) then
FFirstItem:=CurHashItem;
FItems[Index]:=CurHashItem;
if CurHashItem^.Next<>nil then begin
CurHashItem^.Next^.Prior:=CurHashItem;
if CurHashItem^.Prior<>nil then
CurHashItem^.Prior^.Next:=CurHashItem;
end;
CurHashItem:=NextHashItem;
end;
end;
procedure TDynHashArray.SaveCacheItem(Item: Pointer; Index: integer);
// Important:
// !!! Only call this method for items, that exists in the list or for items
// that can't change their key
begin
if Assigned(OnGetKeyForHashItem) then Item:=OnGetKeyForHashItem(Item);
FHashCacheItem:=Item;
FHashCacheIndex:=Index;
end;
constructor TDynHashArray.Create;
begin
Create(10);
end;
procedure TDynHashArray.SetOnGetKeyForHashItem(
const AValue: TOnGetKeyForHashItem);
begin
if FOnGetKeyForHashItem=AValue then exit;
FOnGetKeyForHashItem:=AValue;
RebuildItems;
end;
procedure TDynHashArray.SetOptions(const AValue: TDynHashArrayOptions);
begin
if FOptions=AValue then exit;
FOptions:=AValue;
if (FContainsCache<>nil) xor (dhaoCacheContains in Options) then begin
if FContainsCache=nil then begin
FContainsCache:=TRecentList.Create(5);
end else begin
FContainsCache.Free;
FContainsCache:=nil;
end;
end;
end;
{ TDynHashArrayItemMemManager }
procedure TDynHashArrayItemMemManager.SetMaxFreeRatio(NewValue: integer);
begin
if NewValue<0 then NewValue:=0;
if NewValue=FMaxFreeRatio then exit;
FMaxFreeRatio:=NewValue;
end;
procedure TDynHashArrayItemMemManager.SetMinFree(NewValue: integer);
begin
if NewValue<0 then NewValue:=0;
if NewValue=FMinFree then exit;
FMinFree:=NewValue;
end;
procedure TDynHashArrayItemMemManager.DisposeFirstFreeItem;
var OldItem: PDynHashArrayItem;
begin
if FFirstFree=nil then exit;
OldItem:=FFirstFree;
FFirstFree:=OldItem^.Next;
if FFirstFree<>nil then
FFirstFree^.Prior:=nil;
Dispose(OldItem);
dec(FFreeCount);
end;
procedure TDynHashArrayItemMemManager.DisposeItem(
ADynHashArrayItem: PDynHashArrayItem);
begin
if ADynHashArrayItem=nil then exit;
// unbind item
if ADynHashArrayItem^.Next<>nil then
ADynHashArrayItem^.Next^.Prior:=ADynHashArrayItem^.Prior;
if ADynHashArrayItem^.Prior<>nil then
ADynHashArrayItem^.Prior^.Next:=ADynHashArrayItem^.Next;
// add to free list
ADynHashArrayItem^.Next:=FFirstFree;
FFirstFree:=ADynHashArrayItem;
if ADynHashArrayItem^.Next<>nil then
ADynHashArrayItem^.Next^.Prior:=ADynHashArrayItem;
ADynHashArrayItem^.Prior:=nil;
inc(FFreeCount);
// reduce free list
if (FFreeCount>(((8+FMaxFreeRatio)*FCount) shr 3)) and (FFreeCount>10) then
begin
DisposeFirstFreeItem;
DisposeFirstFreeItem;
end;
end;
function TDynHashArrayItemMemManager.NewItem: PDynHashArrayItem;
begin
if FFirstFree<>nil then begin
Result:=FFirstFree;
FFirstFree:=FFirstFree^.Next;
if FFirstFree<>nil then
FFirstFree^.Prior:=nil;
dec(FFreeCount);
end else begin
New(Result);
end;
with Result^ do begin
Item:=nil;
Next:=nil;
Prior:=nil;
IsOverflow:=false;
end;
end;
procedure TDynHashArrayItemMemManager.Clear;
begin
while FFreeCount>0 do DisposeFirstFreeItem;
end;
constructor TDynHashArrayItemMemManager.Create;
begin
inherited Create;
FFirstFree:=nil;
FFreeCount:=0;
FCount:=0;
FMinFree:=100;
FMaxFreeRatio:=8; // 1:1
end;
destructor TDynHashArrayItemMemManager.Destroy;
begin
Clear;
inherited Destroy;
end;
function TDynHashArrayItemMemManager.ConsistencyCheck: integer;
var RealFreeCount: integer;
HashItem: PDynHashArrayItem;
begin
RealFreeCount:=0;
HashItem:=FFirstFree;
while HashItem<>nil do begin
inc(RealFreeCount);
if (HashItem^.Next<>nil) and (HashItem^.Next^.Prior<>HashItem) then
exit(-1);
if (HashItem^.Prior<>nil) and (HashItem^.Prior^.Next<>HashItem) then
exit(-2);
HashItem:=HashItem^.Next;
end;
if RealFreeCount<>FFreeCount then exit(-3);
Result:=0;
end;
procedure TDynHashArrayItemMemManager.WriteDebugReport;
begin
DebugLn('TDynHashArrayItemMemManager.WriteDebugReport:'
,' Consistency=',dbgs(ConsistencyCheck),', FreeCount=',dbgs(FFreeCount));
end;
//==============================================================================
finalization
FreeAndNil(ItemMemManager);
end.
|