AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Mit BlockRead() Datei kopieren

Ein Thema von sk0r · begonnen am 11. Sep 2007 · letzter Beitrag vom 14. Sep 2007
Antwort Antwort
Seite 1 von 2  1 2      
sk0r

Registriert seit: 1. Mai 2007
181 Beiträge
 
Delphi 7 Enterprise
 
#1

Mit BlockRead() Datei kopieren

  Alt 11. Sep 2007, 21:16
Hi,

ich möchte eine Datei einlesen und Stück für Stück
wieder, als eine Kopie, zusammensetzen. Dies habe
ich auch schon geschafft, indem ich immer ein Byte
kopiert habe. Das dauert aber viel zu lange. Desshalb
möchte ich gerne 1024 Bytes kopieren.

Am Ende möchte ich nämlich die Bytes von einem
Server zu einem Klienten senden, quasi als Download-Feature,
dass der Client Dateien vom Server herunterladen kann.

Leider schaffe ich nicht mal das bloße, stückweise Kopieren. -.-
Ich bekomme immer einen E/A - Fehler 1784.

Mein Code:
Delphi-Quellcode:
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls;

type
  TForm1 = class(TForm)
    Memo1: TMemo;
    procedure FormCreate(Sender: TObject);
  private
    { Private-Deklarationen }
  public
    { Public-Deklarationen }
  end;

type TDLFile = array[0..1024-1] of Byte;

var
  Form1: TForm1;
  tmpReplace: String = 'C:\test.exe';
  tmpKopie: String = 'C:\test_copy.exe';
  sizeDlDat: Cardinal = 0;
  dlCounter: LongInt = 0;

implementation

{$R *.dfm}

function SizeOfFile(lpFileStr: String):Cardinal;
var
  hDat: File of Byte;
begin
  result := 0;
  if FileExists(lpFileStr) then
  begin
    AssignFile(hDat, lpFileStr);
    ReSet(hDat);
    result := FileSize(hDat);
    CloseFile(hDat);
  end;
end;

procedure WriteIntoDownloadedDat(bByteDat: TDLFile);
var
  hDat: File;
  g: LongInt;
begin
  if FileExists(tmpKopie) then
  begin
    AssignFile(hDat, tmpKopie);
    ReSet(hDat, 1024); //Öffnen, falls sie existiert.
    Seek(hDat, soFromEnd); //An das Ende der Datei springen
    BlockWrite(hDat, bByteDat, sizeof(bByteDat)); //Die neuen 1024 Bytes schreiben. Hier E/A - Fehler 1784.
    CloseFile(hDat); //Schliessen
  end
  else begin
    AssignFile(hDat, tmpKopie);
    ReWrite(hDat, 1); //Datei neu erstellen, falls sie nicht existiert.
    BlockWrite(hDat, bByteDat, sizeof(bByteDat)); //Die ersten 1024 Bytes schreiben
    CloseFile(hDat); //Schliessen
  end;
end;

procedure TForm1.FormCreate(Sender: TObject);
var
  hDlDat: File;
  bDlDat: TDLFile;
  g: Integer;
begin
  memo1.Clear;
  if FileExists(tmpReplace) then
  begin
    sizeDlDat := SizeOfFile(tmpReplace)-1; //Dateigröße - 1
    memo1.Lines.Add('Full size: ' + inttostr(sizeDlDat) + ' bytes');
    AssignFile(hDlDat, tmpReplace); //Dateihandle erstellen
    ReSet(hDlDat, 1024); //Datei öffnen (1024 bytes)
    while not eof(hDlDat) do //Solange wir nicht am Ende der Datei sind..
    begin
      BlockRead(hDlDat, bDlDat[0], sizeof(bDlDat[0])); //1024 Bytes einlesen
      WriteIntoDownloadedDat(bDlDat); //Prozedur aufrufen und die Bytes übergeben.
      Seek(hDlDat, soFromCurrent); //Die 1024 kopierten Bytes überspringen, um zu den nächsten 1024 Bytes zu kommen
      Application.ProcessMessages;
    end;
    CloseFile(hDlDat); //Datei schliessen.
    sizeDlDat := 0;
    tmpReplace := '';
    memo1.Lines.Add('FINISHED!');
  end;
end;

end.
Ich weiß nicht mehr weiter.
Hoffe, ihr könnt mir helfen.
  Mit Zitat antworten Zitat
pstruh
(Gast)

n/a Beiträge
 
#2

Re: Mit BlockRead() Datei kopieren

  Alt 11. Sep 2007, 21:43
Ohne auf deinen Code einzugehen, ich habe da folgende Routine:
Delphi-Quellcode:
procedure FastCopy_All(SouName,DesName:String;DelSou:Boolean);
const BufferSize = 32768;
var Sou,Des : file;
      Buffer : Array[1..BufferSize] of Byte;
      NumRead,
      NumSave : Integer;
begin
 CopiedBytes:=0;

 AssignFile(Sou,SouName); {$I-} reset(Sou,1); {$I+}
 FastCopyResult:=IOresult;
 if FastCopyResult<>0 then exit;

 AssignFile(Des,DesName); {$I-} rewrite(Des,1); {$I+}
 FastCopyResult:=IOresult;
 if FastCopyResult<>0 then exit;

 repeat
  BlockRead (Sou,Buffer,SizeOf(Buffer),NumRead);
  BlockWrite(Des,Buffer,NumRead, NumSave);
  inc(CopiedBytes,NumSave);
  Application.ProcessMessages;
  until (NumRead=0) or (NumSave<>NumRead);

 CloseFile(Sou);
 CloseFile(Des);

 if DelSou then erase(Sou);
end;
Der Procedurkopf sollte eigentlich alles erklären. Die globale Variable FastCopyResult ist global deklariert, das würde ich heute anders machen ...
  Mit Zitat antworten Zitat
jbg

Registriert seit: 12. Jun 2002
3.481 Beiträge
 
Delphi 10.1 Berlin Professional
 
#3

Re: Mit BlockRead() Datei kopieren

  Alt 11. Sep 2007, 21:59
Zitat von sk0r:
ReSet(hDat, 1024); //Öffnen, falls sie existiert.
Damit öffnest du die Datei mit einer Blockgröße von 1024. Das bedeutet, dass der Dritte Parameter bei BlockWrite in Einheiten von 1024 Bytes angegeben wird. Also bei BlockWrite(hDat, bByteDat, 1) werden 1024 Bytes kopiert, bei BlockWrite(hDat, bByteDat, 2) werden 2048 Bytes kopiert usw.

Was du brauchst ist eine Blockgröße von 1. Also Reset(hDat, 1); Dann entspricht der dritte Parameter einer Einheit von einem Byte.

Aber warum nimmst du nicht gleich einen TFileStream her die die alten TurboPascal Funktionen ersetzen.
  Mit Zitat antworten Zitat
sk0r

Registriert seit: 1. Mai 2007
181 Beiträge
 
Delphi 7 Enterprise
 
#4

Re: Mit BlockRead() Datei kopieren

  Alt 12. Sep 2007, 12:45
Danke, habe es nun hinbekommen, obwohl er ein paar Bytes zuviel schreibt. :p
  Mit Zitat antworten Zitat
Benutzerbild von Sunlight7
Sunlight7

Registriert seit: 17. Sep 2006
Ort: Sonnensystem, Zentral
1.522 Beiträge
 
Delphi 5 Standard
 
#5

Re: Mit BlockRead() Datei kopieren

  Alt 13. Sep 2007, 00:34
Hätte man in die Delphi Hilfe geguckt...
Beispiel aus der Delphi Hilfe

Delphi-Quellcode:
var FromF, ToF: file;
  NumRead, NumWritten: Integer;
  Buf: array[1..2048] of Char;
begin
  if OpenDialog1.Execute then { Dialog zum Dateiöffnen anzeigen }
  begin
    AssignFile(FromF, OpenDialog1.FileName);
    Reset(FromF, 1);   { Datensatzgröße = 1 }
    if SaveDialog1.Execute then { Dialog zum Speichern anzeigen }
    begin
      AssignFile(ToF, SaveDialog1.FileName);   { Ausgabedatei öffnen }

      Rewrite(ToF, 1);   { Datensatzgröße = 1 }
      Canvas.TextOut(10, 10, 'Kopieren von ' + IntToStr(FileSize(FromF))
        + ' Byte...');
      repeat
        BlockRead(FromF, Buf, SizeOf(Buf), NumRead);
        BlockWrite(ToF, Buf, NumRead, NumWritten);
      until (NumRead = 0) or (NumWritten <> NumRead);
        CloseFile(FromF);
        CloseFile(ToF);
    end;
  end;
end;


Dann wird nix zuviel kopiert^^
Windows: Ja - Microsoft: Nein -> www.ReactOS.org
  Mit Zitat antworten Zitat
marabu

Registriert seit: 6. Apr 2005
10.109 Beiträge
 
#6

Re: Mit BlockRead() Datei kopieren

  Alt 13. Sep 2007, 06:30
Hallo,

bei BlockRead muss die Dateigröße immer ein Vielfaches der Blockgröße sein - deshalb ist BlockSize=1 irgendwie optimal. Mit Streams reduziert sich der Code eigentlich auf die Methode CopyFrom():

Delphi-Quellcode:
procedure FileCopy(fnSource, fnTarget: TFilename);
var
  fsSource, fsTarget: TStream;
begin
  fsSource := TFileStream.Create(fnSource, fmOpenRead);
  try
    fsTarget := TFileStream.Create(fnTarget, fmCreate);
    try
      fsTarget.CopyFrom(fsSource, 0);
    finally
      fsTarget.Free;
    end;
  finally
    fsSource.Free;
  end;
end;
Getippt und nicht getestet.

Grüße vom marabu
  Mit Zitat antworten Zitat
sk0r

Registriert seit: 1. Mai 2007
181 Beiträge
 
Delphi 7 Enterprise
 
#7

Re: Mit BlockRead() Datei kopieren

  Alt 13. Sep 2007, 12:24
Ich schaue immer in die Delphi Hilfe, aber diesmal konnte ich nichts finden.
Sonst würde ich ja nicht fragen.

Mein Code sieht im Moment so aus:

Delphi-Quellcode:
      if FileExists(tmpReplace) then
      begin
        sizeDlDat := SizeOfFile(tmpReplace)-1;
        AssignFile(hDlDat, tmpReplace);
        ReSet(hDlDat, 1);
        g := 0;
        repeat
          BlockRead(hDlDat, bDlDat, 1024);
          Socket.SendBuf(bDlDat, 1024);
          inc(g, 1024);
          Seek(hDlDat, soFromBeginning+g);
          Sleep(7);
          Application.ProcessMessages;
        until g >= (sizeDlDat-1024);
        CloseFile(hDlDat);
        sizeDlDat := 0;
      end;
      tmpReplace := '';
Wenn ich anstatt das Socket.SendBuf WriteIntoDownloadedDat() benutze, dann klappt es.
Nur leider sendet er immer ganz komische Bytes. Da weiß ich leider nicht mehr weiter.
  Mit Zitat antworten Zitat
marabu

Registriert seit: 6. Apr 2005
10.109 Beiträge
 
#8

Re: Mit BlockRead() Datei kopieren

  Alt 13. Sep 2007, 13:12
Hallo,

du lässt BlockRead() immer 1024 Byte lesen, aber beim letzten Lesevorgang ist die Chance groß, dass weniger als 1024 Byte gelesen werden können. Du kannst die Zahl der wirklich gelesenen Bytes in einer Variablen erhalten, die du als vierten Parameter von BlockRead() übergibst. Diese Variable kannst du dann zum Steuern von SendBuf() verwenden, damit nicht mehr als die gelesenen Bytes versendet werden.

Delphi-Quellcode:
// look ma, no hands!

procedure FileSend(const fn: TFilename; socket: TBaseSocket);
var
  s: TStream;
begin
  s := TFileStream.Create(fn, fmOpenRead);
  try
    socket.SendStream(s);
  finally
    s.Free;
  end;
end;
Wieder getippt und nicht getestet.

Freundliche Grüße
  Mit Zitat antworten Zitat
sk0r

Registriert seit: 1. Mai 2007
181 Beiträge
 
Delphi 7 Enterprise
 
#9

Re: Mit BlockRead() Datei kopieren

  Alt 13. Sep 2007, 14:36
Naja, das Problem ist im Moment, dass die gedownloadete Datei sich nicht im geringsten
mit der originalen Datei mehr ähnelt, wenn ich sie abgleiche. Die Kopie des Originals
hat immer das selbe Muster, also z.B. so:

++++++++++++
++++
++++++++++++
++++
++++++++++++
++++
++++++++++++
++++

Die originale Datei dagegen hat das richtige Muster.
Auch enthält die kopierte Datei nicht mehr die Strings, welche die originale Datei enthält.

Das war jetzt bisschen doof zu erklären. Ich hoffe ihr versteht, was ich meine. :/
  Mit Zitat antworten Zitat
sk0r

Registriert seit: 1. Mai 2007
181 Beiträge
 
Delphi 7 Enterprise
 
#10

Re: Mit BlockRead() Datei kopieren

  Alt 13. Sep 2007, 14:37
Zitat von sk0r:
Naja, das Problem ist im Moment, dass die gedownloadete Datei sich nicht im geringsten
mit der originalen Datei mehr ähnelt, wenn ich sie abgleiche. Die Kopie des Originals
hat immer das selbe Muster, also z.B. so:

++++++++++++
++++
++++++++++++
++++
++++++++++++
++++
++++++++++++
++++

Die originale Datei dagegen hat das richtige Muster.
Auch enthält die kopierte Datei nicht mehr die Strings, welche die originale Datei enthält.

Das war jetzt ein bisschen schwer zu erklären. Ich hoffe ihr versteht, was ich meine. :/
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 12:05 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