我有一个本机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#code
StringBuilder b = new StringBuilder(1000);
getDetails(b, 255);
m = b.ToString(); //always ""
Console.WriteLine(m);
当我单步执行代码时,我看到值(lpDetails)被设置为正确的值。问题出在c#上,我看到的只是一个空字符串。
我尝试过从其他帖子中找到的东西,但到目前为止还没有运气。
答案 0 :(得分:1)
这里有一些问题:
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
}_