Delphi中如何将文件读入字节数组?

Ken*_*ker 4 arrays delphi byte delphi-xe2

我正在尝试将文件读入 Delphi XE2 中的字节数组。

这是我当前的代码:

function FileToBytes(const AName: string; var Bytes: TBytes): Boolean;
var
  Ms: TMemoryStream;
begin
  Result := False;
  if not FileExists(AName) then
    Exit;
  Ms := TMemoryStream.Create;
  try
    Ms.LoadFromFile(AName);
    if Ms.Size > 0 then
    begin
      Ms.Position := 0;
      MS.ReadBuffer(Bytes[0], Ms.Size);
      Result := True;
    end;
  finally
    Ms.Free;
  end;
end;

procedure runFile();
var
  Bytes: TBytes;
  OpFile: String;
begin
  OpFile := 'C:\Users\Kenny\Documents\calc.exe';
  Bytes := nil;
  if FileToBytes(OpFile, Bytes) then
  begin
    //do someting with Bytes(array of Byte)

  end;
end;
Run Code Online (Sandbox Code Playgroud)

我在这一行收到错误:

MS.ReadBuffer(Bytes[0], Ms.Size);
Run Code Online (Sandbox Code Playgroud)

错误是:

0x00404727 处的访问冲突:写入地址 0x00000008

任何解决此问题的帮助将不胜感激。

Dav*_*nan 5

您没有分配数组,这解释了错误。您可以像这样修复您的代码:

Ms.LoadFromFile(AName);
SetLength(Bytes, Ms.Size);
Ms.Position := 0;
MS.ReadBuffer(Pointer(Bytes)^, Ms.Size);
Result := True;
Run Code Online (Sandbox Code Playgroud)

请注意,我已经避免了检查零长度文件的需要,并使用了这样的代码Pointer(Bytes),以便在范围检查处于活动状态时代码可以工作。

我还要指出的是,您的代码违反了我所说的 Delphi 内存流反模式。您将文件读入内存流,它本质上是一个字节数组。然后从该字节数组复制到另一个字节数组。您将整个文件写入两个单独的字节数组。这比必要的多了一个。最好这样写:

function FileToBytes(const AName: string; var Bytes: TBytes): Boolean;
var
  Stream: TFileStream;
begin
  if not FileExists(AName) then
  begin
    Result := False;
    Exit;
  end;
  Stream := TFileStream.Create(AName, fmOpenRead);
  try
    SetLength(Bytes, Stream.Size);
    Stream.ReadBuffer(Pointer(Bytes)^, Stream.Size);
  finally
    Stream.Free;
  end;
  Result := True;
end;
Run Code Online (Sandbox Code Playgroud)

这样您就可以直接从文件读入目标字节数组。

我不太喜欢将其作为返回布尔值以指示成功的函数。除了文件不存在之外,还有很多方式会导致代码失败。这些都会导致您的设计出现异常。我更愿意看到纯粹基于异常的错误处理,或纯粹基于错误代码的错误处理。但不是你的设计的混合方法。

正如 Andreas 在评论中指出的那样,您还可以使用 RTL 库函数System.IOUtils.TFile.ReadAllBytes来执行此任务。System.IOUtils尽管就我个人而言,由于许多设计问题,我倾向于避免使用整个单元。虽然其中一些问题已在最近的版本中得到解决,但我相信 XE2 版本可能会存在一些问题。