试图从本机DLL获取字符串值

时间:2012-09-12 04:10:14

标签: c# c++ pinvoke

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

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

c ++方法

我已经硬编码返回lpDetails = L“test”;

我的c#签名

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

c#c​​ode

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

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

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

1 个答案:

答案 0 :(得分:1)

这里有一些问题:

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

即使没有Interop,这也行不通:

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
}_