0

我有一个本机 dll,我需要设置参数的字符串值。下面是 C++ 方法签名

  __declspec(dllexport) void __stdcall getDetails(_Out_ LPTSTR lpDetails, _In_ UINT uSize)

c++方法

我已经为 return lpDetails=L"test"; 硬编码了这个。

我的 c# 签名

   [return: MarshalAs(UnmanagedType.I1)]
   [DllImport("test.dll",CharSet=CharSet.Auto)]
   static extern bool getDetails(StringBuilder result, System.UInt32 uSize);

c# 代码

        StringBuilder b = new StringBuilder(1000);
        getDetails(b, 255);
        m = b.ToString(); //always ""
        Console.WriteLine(m);

当我单步执行代码时,我看到值 (lpDetails) 被设置为正确的值。问题出在 c# 上,我所看到的只是一个空字符串。

我尝试了从其他帖子中找到的东西,但到目前为止还没有运气。

4

1 回答 1

1

这里有几件事是错误的:

  • 本机方法显示为 C 签名,而不是 C++ 签名(我在这里挑剔:)
  • lPDetails 参数是 IN,而不是 OUT,如果您希望函数将详细信息写入预分配的位置(从第二个参数看来,它应该表示分配的缓冲区的长度)
  • 你的实现lpDetails=L"test"被破坏了。如果您向函数提供预分配的缓冲区(正如您的签名所暗示的那样),您需要_tcscpy将详细信息放入 lpDetails。如果您想在函数内分配字符串(正如您的实现所暗示的,您需要将 lpDetails 声明为LPTSTR*,但这是一个糟糕的设计,因为调用者现在可能不知道如何释放它(在这种情况下,它确实不能作为“测试”指向数据段中的静态内存而不是堆。

即使没有互操作,这也行不通:

LPTSTR pCallerData; // a pointer variable
void getDetails(LPTSTR lpData, /*what do you use second param for? */)
{
  // lpData is a COPY of pCallerData, not a pointer to it, 
  // so you are not modifying pCallerData, but a local variable
  lpData = "test";
  // after the function exits pCallerData still has the same value as before
}_

更好的实施将是:

LPTSTR txt[256]; 
getDetails(txt, 256);
// ----
void getDetails(LPTSTR lpData, int size)
{
_tcscpy(lpDetails, _T("Test")); 
// or more realistically _tcsncpy and properly check 
// if details string is longer than size argument
}_
于 2012-09-12T07:49:17.827 回答