Delphi-PRAXiS
Seite 2 von 2     12   

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   Binäre (Hex) Suche (https://www.delphipraxis.net/173782-binaere-hex-suche.html)

lbccaleb 16. Mär 2013 13:23

AW: Binäre (Hex) Suche
 
Zitat:

Zitat von Aphton (Beitrag 1207627)
... Falls Interesse besteht, kann ich die kleine Demo ja hochladen!

Also ich würd mal Intresse anmelden :)

Alter Mann 16. Mär 2013 14:08

AW: Binäre (Hex) Suche
 
Hi,

ich habe ein halbwegs brauchbares Ergebnis. Die Frage ist nun: geht das auch schneller?
Delphi-Quellcode:
unit uHexSearch;

interface

uses
  SysUtils, Classes;

type
  THexSearchFoundEvent = procedure(Sender    : TObject;
                                    Position  : Integer;
                                    var Cancel : Boolean) of object;
  THexSearchProgressEvent = procedure(Sender  : TObject;
                                      Position : WORD;
                                      Max     : Int64) of Object;

  THexByteArray        = Array of Byte;

  TFiFoByteBuffer      = class(TObject)
  private
    FBuffer            : THexByteArray;
    FCount             : Int64;
    function   GetHasData : Boolean;
    function   GetByteStr : String;
  public
    constructor Create(aSize : Integer);
    procedure  Add(Value : Byte);
    property   HasData : Boolean read GetHasData;
    property   ByteStr : String read GetByteStr;
  end;

  THexSearch           = class(TComponent)
  private
    FBytes             : String;
    FBuffer            : TFiFoByteBuffer;
    FCanCancel         : Boolean;
    FStream            : TStream;
    FSearchTime        : TDateTime;
    FOnFound           : THexSearchFoundEvent;
    FOnProgress        : THexSearchProgressEvent;
    procedure Search(aStream     : TStream;
                     const aBytes : String);
    function EqualBytes(aSource, aSearchStr : String) : Boolean;
  protected
    procedure DoFound(Position : Integer; var Cancel : Boolean); virtual;
    procedure DoProgress(Position : WORD; Max : Int64); virtual;
  public
    procedure Execute;

    property Cancel     : Boolean   read FCanCancel write FCanCancel;
    property Stream     : TStream   read FStream   write FStream;
    property SearchBytes : string    read FBytes    write FBytes;
    property SearchTime : TDateTime read FSearchTime;
  published
    property OnFound    : THexSearchFoundEvent   read FOnFound   write FOnFound;
    property OnProgress : THexSearchProgressEvent read FOnProgress write FOnProgress;
  end;

implementation

constructor TFiFoByteBuffer.Create(aSize : Integer);
begin
  SetLength(FBuffer, aSize);
  FCount := 0;
end;

procedure  TFiFoByteBuffer.Add(Value : Byte);
var
  I : WORD;
begin
  for I := Low(FBuffer) to High(FBuffer) -1 do
   FBuffer[I] := FBuffer[I+1];
  FBuffer[High(FBuffer)] := Value;
  Inc(FCount);
end;

function   TFiFoByteBuffer.GetHasData : Boolean;
begin
  Result := FCount >= Length(FBuffer);
end;

function   TFiFoByteBuffer.GetByteStr : String;
var
  I : Integer;
begin
  Result := '';
  for I := 0 to Length(FBuffer)-1 do Result := Result + IntToHex(FBuffer[I], 2);
end;

procedure THexSearch.DoFound(Position : Integer; var Cancel : Boolean);
begin
  If Assigned(FOnFound) then FOnFound(Self,Position, Cancel);
end;

procedure THexSearch.DoProgress(Position : WORD; Max : Int64);
begin
  If Assigned(FOnProgress) then FOnProgress(Self,Position, Max);
end;

procedure THexSearch.Execute;
begin
  FSearchTime := Now;
  Search(FStream, FBytes);
  FSearchTime := Now - FSearchTime;
end;

procedure THexSearch.Search(aStream     : TStream;
                            const aBytes : String);
var
  Buffer     : Byte;
  FileSize   : Int64;
begin
  FCanCancel := false;
  FileSize   := aStream.Size;
  FBuffer    := TFiFoByteBuffer.Create(Length(aBytes) div 2);
  try
  aStream.Seek(0, soFromBeginning);
    While (aStream.Position < aStream.Size) do
    begin
      aStream.Read(Buffer, SizeOf(Buffer));
      FBuffer.Add(Buffer);
      DoProgress(aStream.Position, FileSize);
      if FBuffer.HasData then
        if EqualBytes(FBuffer.ByteStr, aBytes) then
          DoFound(aStream.Position, FCanCancel);
      if FCanCancel then Break;
    end;
  finally
    FBuffer.Free;
  end;
end;

function THexSearch.EqualBytes(aSource, aSearchStr : String) : Boolean;
begin
  Result := CompareText(aSource, aSearchStr) = 0;
end;

end.
Das durchsuchen einer 5GB dauert 0.328 sec, bzw. nach dieser Zeit ist die
Bytefolge gefunden worden.

VG

Aphton 16. Mär 2013 14:33

AW: Binäre (Hex) Suche
 
Liste der Anhänge anzeigen (Anzahl: 1)
Zitat:

Zitat von Alter Mann (Beitrag 1207691)
Das durchsuchen einer 5GB dauert 0.328 sec, bzw. nach dieser Zeit ist die
Bytefolge gefunden worden.

VG

Das sagt nicht viel aus. Wichtig ist, an welcher Stelle sie gefunden wird. Dann kann man nämlich sagen, wie viele Bytes pro Zeiteinheit du damit durchkämmen kannst!

Die Geschwindigkeit dürfte übringens auch von Festplatte zu Festplatte verschieden sein. Eig. kann man nur lokal bzw. angewandt auf derselben HD Vergleiche aufstellen!

Achja - ich weiß nicht, ob man da noch was rausholen kann oder ob das eig. Unfug ist - so wie ich mit CreateFileMapping() umgegangen bin. Ich erfreue mich jedoch über Belehrung!

Edit: Ich wende mal deinen Algorithmus an, mal sehen, wenn er schneller ist, brauchste die Demo erst gar nicht runterzuladen!

Edit2: Dein Algorithmus läuft bei mir mit einer sehr geringen Geschwindigkeit. Für 1 mb habe ich 5 Sek. gebraucht :(
Ich wende sie übrigens auf ein FileStream an (genauso wie ich es auch bei meinem getan habe).
Wie hast du die angegebene Zeit Zustande gebracht?


Alle Zeitangaben in WEZ +1. Es ist jetzt 22:53 Uhr.
Seite 2 von 2     12   

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