Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Neuen Beitrag zur Code-Library hinzufügen (https://www.delphipraxis.net/33-neuen-beitrag-zur-code-library-hinzufuegen/)
-   -   Delphi Zufällig Zahlen ziehen, aber ohne Wiederholung (https://www.delphipraxis.net/150896-zufaellig-zahlen-ziehen-aber-ohne-wiederholung.html)

alzaimar 29. Apr 2010 19:24


Zufällig Zahlen ziehen, aber ohne Wiederholung
 
Hier eine Klasse, die zufällig Zahlen aus einem Zahlenbereich zieht, bis keine mehr übrig sind. Jede Zahl wird nur 1x gezogen. Die Klasse erstellt eine Liste der Zahlen, mischt sie nach Fisher-Yates und liefert dann eine nach dem anderen zurück.

Verwendung (Lotto, 6 aus 49 + Zusatzzahl);
Delphi-Quellcode:
With TUniqueRandomSequenceGenerator.Create (1,49) do begin
  Write('Und hier die Lottozahlen: ');
  For i := 1 to 6 do
    Write(GetNextNumber:2,' ');
  Writeln(' Zusatzzahl ',GetNextNumber);
  Free
End;
Und hier die Klasse:
Delphi-Quellcode:
type
  ENoMoreNumbers = Exception;
  TUniqueRandomSequenceGenerator = class
  private
    FCounter: Integer;
    FNumberList: array of Integer;
    procedure BuildSequence(aStart, aEnde: Integer);
  public
    constructor Create(aStart, aEnde: Integer);
    function GetNextNumber: Integer;
    function TotalCount: Integer;
    function RemainingCount: Integer;
  end;
implementation

{ TUniqueRandomGenerator }

constructor TUniqueRandomSequenceGenerator.Create(aStart, aEnde: Integer);
begin
  BuildSequence(aStart, aEnde);
end;

procedure TUniqueRandomSequenceGenerator.BuildSequence(aStart, aEnde: Integer);
var
  i, j, tmp: Integer;

begin
  SetLength(FNumberList, aEnde - aStart + 1);
// Zahlenliste erzeugen
  for i := 0 to TotalCount - 1 do
    FNumberList[i] := aStart + i;

// Mischen nach Fisher-Yates
  for i := Low(FNumberList) to High(FNumberList) do begin
    j := i + Random(TotalCount - i);
    tmp := FNumberList[j];
    FNumberList[j] := FNumberList[i];
    FNumberList[i] := tmp;
  end;
  FCounter := 0;
end;

function TUniqueRandomSequenceGenerator.GetNextNumber: Integer;
begin
  if FCounter < TotslCount then begin
    Result := FNumberList[FCounter];
    Inc(FCounter);
  end
  else
    raise ENoMoreNumbers.Create('No more numbers');
end;

function TUniqueRandomSequenceGenerator.RemainingCount: Integer;
begin
  Result := TotalCount - FCounter;
end;

function TUniqueRandomSequenceGenerator.TotalCount: Integer;
begin
  Result := Length(FNumberList);
end;
[edit]Schlampenfehler dank gsammatester korrigiert[/edit]

gammatester 29. Apr 2010 20:02

Re: Zufällig Zahlen ziehen, aber ohne Wiederholung
 
Also mit Delphi 2 und/oder 3 kann man's nicht übersetzen, da zB keine dynamischen Arrays bekannt sind.

Dann ein Bug: '<' muss durch '<=" ersetzt werden, sonst erhält man nicht die gesamte Liste (das berüchtigte 1-mehr/weniger-Problem):
Delphi-Quellcode:
function TUniqueRandomSequenceGenerator.GetNextNumber: Integer;
begin
  if FCounter <= High(FNumberList) then begin // hier <= statt <
    Result := FNumberList[FCounter];
    Inc(FCounter);
  end
  else
    raise ENoMoreNumbers.Create('No more numbers');
end;
Aus Konsistenzgünden sollte statt Length(FNumberList) hier
Delphi-Quellcode:
j := i + Random(Length(FNumberList) - i);
auch TotalCount benutzt werden,


Gruß Gammatester

Luckie 29. Apr 2010 20:06

Re: Zufällig Zahlen ziehen, aber ohne Wiederholung
 
Zitat:

Zitat von gammatester
Also mit Delphi 2 und/oder 3 kann man's nicht übersetzen, da zB keine dynamischen Arrays bekannt sind.

Nun ja. Das würde mich aber nicht sonderlich stören. Delphi 3 ist immer hin 13 Jahre alt und Delphi 2 sogar 12.

Klaus01 29. Apr 2010 20:22

Re: Zufällig Zahlen ziehen, aber ohne Wiederholung
 
Zitat:

Zitat von Luckie
Delphi 3 ist immer hin 13 Jahre alt und Delphi 2 sogar 12.

Sorry could not resist

Klaus

alzaimar 30. Apr 2010 05:53

Re: Zufällig Zahlen ziehen, aber ohne Wiederholung
 
Hi gammatester,

wenn Du nicht wärst.... :thumb:


Alle Zeitangaben in WEZ +1. Es ist jetzt 07:16 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz