| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 | {$mode objfpc} {$h+} {$typedaddress on} {$modeswitch advancedrecords} {$coperators on}{$modeswitch anonymousfunctions}{$modeswitch duplicatelocals}uses	{$ifdef unix}cwstring,{$endif}	SysUtils, Generics.Collections, Generics.Defaults;var	anythingWrong: boolean = false;	generic procedure Swap<Ty>(var a, b: Ty);	var		temp: Ty;	begin		temp := a; a := b; b := temp;	end;type	generic TTracingComparer<Ty> = class(specialize TComparer<Ty>)		orig: specialize IComparer<Ty>;		count: uint64;		constructor Create(const orig: specialize IComparer<Ty>);		function Compare(const a, b: Ty): integer; override;	end;	constructor TTracingComparer.Create(const orig: specialize IComparer<Ty>);	begin		inherited Create;		self.orig := orig;	end;	function TTracingComparer.Compare(const a, b: Ty): integer;	begin		result := orig.Compare(a, b);		count += 1;	end;type	// https://igoro.com/archive/quicksort-killer/	// Will work against wide range of qsort implementations.	TQSortKillerComparer = class(specialize TComparer<SizeInt>)		keys: array of int32; { TDictionary is a lot slower... }		candidate, nKeys: int32;		constructor Create(arrayLen: SizeInt);		function Compare(const a, b: SizeInt): integer; override;	end;	constructor TQSortKillerComparer.Create(arrayLen: SizeInt);	begin		inherited Create;		SetLength(keys, arrayLen);		FillChar(pInt32(keys)^, length(keys) * sizeof(keys[0]), byte(-1));	end;	function TQSortKillerComparer.Compare(const a, b: SizeInt): integer;	begin		if keys[a] and keys[b] < 0 then		begin			if a = candidate then keys[a] := nKeys else keys[b] := nKeys;			nKeys += 1;		end;		if keys[a] < 0 then begin candidate := a; exit(1); end;		if keys[b] < 0 then begin candidate := b; exit(-1); end;		result := keys[a] - keys[b];	end;type	generic SortBenchmark<Ty> = record	type		CreateProc = function(id: SizeUint): Ty;		TyArray = array of Ty;		class procedure Run(create: CreateProc; const tyPlural: string; lenMul: double); static;		class procedure BenchSort(const src, ref: array of Ty; var prevComparisons: uint64); static;		// 'ref' must be sorted and contain no duplicates.		class function BuildQSortKiller(const ref: array of Ty): TyArray; static;	end;	class procedure SortBenchmark.Run(create: CreateProc; const tyPlural: string; lenMul: double);	type		OrderEnum = (RandomOrder, QSortKillerOrder);	const		OrderNames: array[OrderEnum] of string = ('random', 'QSort killer');	var		ref, src: TyArray;		i, lenBase, len: SizeInt;		cmp: specialize IComparer<Ty>;		srcOrder: OrderEnum;		msg: string;		prevComparisons: uint64;	begin		writeln('--- ', tyPlural, ' ---', LineEnding);		for srcOrder in OrderEnum do		begin			writeln('Order: ', OrderNames[srcOrder], '.');			prevComparisons := uint64(-1);			for lenBase in specialize TArray<SizeInt>.Create(10 * 1000, 20 * 1000, 40 * 1000) do			begin				len := round(lenMul * lenBase);				SetLength((@ref)^, len);				cmp := specialize TComparer<Ty>.Default;				for i := 0 to len - 1 do				begin					ref[i] := create(i);					if (i > 0) and (cmp.Compare(ref[i - 1], ref[i]) >= 0) then					begin						writeln('''create'' callback must return ', tyPlural, ' in strictly ascending order.');						anythingWrong := true;						exit;					end;				end;				case srcOrder of					RandomOrder:						begin							RandSeed := 1;							src := Copy(ref);							for i := len - 1 downto 1 do								specialize Swap<Ty>(src[i], src[random(int64(i))]);						end;					QSortKillerOrder:						src := BuildQSortKiller(ref);				end;				WriteStr(msg, 'n = ', len, ': ');				write(msg.PadRight(12));				BenchSort(src, ref, prevComparisons);			end;			writeln;		end;	end;	class function SortBenchmark.BuildQSortKiller(const ref: array of Ty): TyArray;	var		ris: array of SizeInt;		i: SizeInt;		cmpRef: specialize IComparer<SizeInt>;	begin		SetLength((@ris)^, length(ref));		for i := 0 to High(ris) do ris[i] := i;		cmpRef := TQSortKillerComparer.Create(length(ref));		specialize TArrayHelper<SizeInt>.Sort(ris, cmpRef);		SetLength((@result)^, length(ref));		for i := 0 to High(result) do result[ris[i]] := ref[i];	end;	class procedure SortBenchmark.BenchSort(const src, ref: array of Ty; var prevComparisons: uint64);	var		arr: TyArray;		i: SizeInt;		cmp: specialize TTracingComparer<Ty>;		cmpRef: specialize IComparer<Ty>;		prevCount: uint64;	begin		cmp := specialize TTracingComparer<Ty>.Create(specialize TComparer<Ty>.Default);		cmpRef := cmp;		arr := Copy(src);		specialize TArrayHelper<Ty>.Sort(arr, cmpRef);		prevCount := prevComparisons;		prevComparisons := cmp.count;		write(cmp.count, ' comparisons');		if prevCount <> uint64(-1) then write(' (', cmp.count / prevCount:0:1, 'x from previous)');		for i := 0 to High(ref) do			if arr[i] <> ref[i] then			begin				writeln(', FAIL @ ', i, ' / ', length(ref));				anythingWrong := true;				exit;			end;		if (prevCount <> uint64(-1)) and (cmp.count > 2 * prevCount + prevCount div 2 + 5 * length(src) + 1000) then		begin			writeln(', potentially bad sorting algorithm behaviour');			anythingWrong := true;			exit;		  end;		writeln(', OK');	end;begin	specialize SortBenchmark<string>.Run(		function(id: SizeUint): string		begin			SetLength((@result)^, 5);			result[5] := Char(ord('A') + id mod 26); id := id div 26;			result[4] := Char(ord('A') + id mod 26); id := id div 26;			result[3] := Char(ord('A') + id mod 26); id := id div 26;			result[2] := Char(ord('A') + id mod 26); id := id div 26;			result[1] := Char(ord('A') + id mod 26);		end, 'strings', 0.15);	specialize SortBenchmark<single>.Run(		function(id: SizeUint): single		begin			result := -1000 + id / 1000;		end, 'float32''s', 1.0);	if anythingWrong then	begin		writeln('Something was wrong, see above.');		halt(2);	end;end.
 |