dmi*_*lam 29 delphi string unicode memorystream delphi-2009
在Delphi 2009之前我们有以下代码:
function MemoryStreamToString(M : TMemoryStream): String;
var
    NewCapacity: Longint;
begin
    if (M.Size = > 0) or (M.Memory = nil) then
       Result:= '' 
    else
    begin
       if TMemoryStreamProtected(M).Capacity = M.Size then
       begin
           NewCapacity:= M.Size+1;
           TMemoryStreamProtected(M).Realloc(NewCapacity);
       end;
       NullString(M.Memory^)[M.Size]:= #0;
       Result:= StrPas(M.Memory);
    end;
end;
我们如何使用Delphi 2009将此代码转换为支持Unicode?
Rob*_*edy 65
您拥有的代码不必要地复杂,即使对于较旧的Delphi版本也是如此.为什么要获取流的字符串版本会强制重新分配流的内存?
function MemoryStreamToString(M: TMemoryStream): string;
begin
  SetString(Result, PChar(M.Memory), M.Size div SizeOf(Char));
end;
这适用于所有Delphi版本,而不仅仅是Delphi 2009.它在流为空时没有任何特殊情况.SetString是一个不被重视的功能.
如果您切换到Delphi 2009时流的内容没有更改为Unicode,那么您应该使用此函数:
function MemoryStreamToString(M: TMemoryStream): AnsiString;
begin
  SetString(Result, PAnsiChar(M.Memory), M.Size);
end;
这相当于您的原始代码,但跳过了特殊情况.
Joh*_*mas 16
或者您可以重构代码以直接使用TStringStream?您可以使用它而不是TMemoryStream(它们具有相同的接口),您只需调用myString:= myStringStream.DataString即可将其"转换"为字符串.
Nic*_*ges 13
一种"更清洁"的方式可能是:
function StreamToString(aStream: TStream): string;
var
  SS: TStringStream;
begin
  if aStream <> nil then
  begin
    SS := TStringStream.Create('');
    try
      SS.CopyFrom(aStream, 0);  // No need to position at 0 nor provide size
      Result := SS.DataString;
    finally
      SS.Free;
    end;
  end else
  begin
    Result := '';
  end;
end;
我用:
function StreamToString(const Stream: TStream; const Encoding: TEncoding): string;
var
  StringBytes: TBytes;
begin
  Stream.Position := 0;
  SetLength(StringBytes, Stream.Size);
  Stream.ReadBuffer(StringBytes, Stream.Size);
  Result := Encoding.GetString(StringBytes);
end;
它仅在 Delphi XE7 上进行了测试。