我有一个DLL,其中我有一个返回pchar的函数. (为了避免使用borlndmm)我最初做的是将一个字符串作为pchar转换并返回
Result := pChar(SomeFuncThatReturnsString)
但是我在90%的时间里获得了预期的结果,而其他时候我什么也得不到.
然后我开始认为我需要为pchar分配内存,并且我原来的方式就是拥有一个pchar指向内存,这并不总是在最初调用该函数时的内容.所以我现在有这个
Result := StrAlloc(128); Strcopy(Result,PAnsiChar(Hash(Hash(Code,1,128),2,128)));
但这让我不得不清理程序端的已分配内存
StrDispose(Pstr);
所以64美元的问题是:从DLL中的函数返回PChar时我是否必须分配内存,或者我可以将其转换为PChar?
这个问题的典型方法是让应用程序分配内存然后将其传递给DLL来填充(如果DLL允许应用程序查询需要分配多少内存,那么它就更好了,因此它不必覆盖 – 分配内存):
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;
这允许应用程序决定何时以及如何分配内存(堆栈与堆,重用内存块等):
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;
如果这不适合您,那么您需要让DLL分配内存,将其返回给应用程序使用,然后让DLL导出一个额外的函数,当完成传递指针时应用程序可以调用该函数回到DLL中解放:
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;