In Delphi in my DLL do I have to allocate the return pchar of a function

后端 未结 3 509
长情又很酷
长情又很酷 2021-02-14 09:20

I have a DLL in which I have a function that returns a pchar. (as to avoid having to use borlndmm) What I was doing originally was casting a string as a pchar and returning that

3条回答
  •  夕颜
    夕颜 (楼主)
    2021-02-14 10:02

    The typical approach to this issue is to have the app allocate the memory and then pass it to the DLL to fill in (even better if the DLL allows the app to query how much memory it needs to allocate so it does not have to over-allocate memory):

    function GetAString(Buffer: PChar; BufLen: Integer): Integer; stdcall;
    var
      S: String;
    begin
      S := SomeFuncThatReturnsString;
      Result := Min(BufLen, Length(S));
      if (Buffer <> nil) and (Result > 0) then
        Move(S[1], Buffer^, Result * SizeOf(Char));
    end;
    

    This allows the app to decide when and how to allocate the memory (stack versus heap, reusing memory blocks, etc):

    var
      S: String;
    begin
      SetLength(S, 256);
      SetLength(S, GetAString(PChar(S), 256));
      ...
    end;
    
    var
      S: String;
    begin
      SetLength(S, GetAString(nil, 0));
      if Length(S) > 0 then GetAString(PChar(S), Length(S));
      ...
    end;
    
    var
      S: array[0..255] of Char;
      Len: Integer;
    begin
      Len := GetAString(S, 256);
      ...
    end;
    

    If this is not an option for you, then you need to have the DLL allocate the memory, return it to the app for use, and then have the DLL export an additional function that the app can call when it is done to pass the pointer back to the DLL for freeing:

    function GetAString: PChar; stdcall;
    var
      S: String;
    begin
      S := SomeFuncThatReturnsString;
      if S <> '' then
      begin
        Result := StrAlloc(Length(S)+1);
        StrPCopy(Result, S);
      end else
        Result := nil;
    end;
    
    procedure FreeAString(AStr: PChar); stdcall;
    begin
      StrDispose(AStr);
    end;
    
    var
      S: PChar;
    begin
      S := GetAString;
      if S <> nil then
      try
        ...
      finally
        FreeAString(S);
      end;
    end;
    

提交回复
热议问题