我已经阅读了其他类似的问题,但他们没有解决这个特殊的问题。我有一个带有touppercase功能的旧C库(作为示例)。这需要一个char *并返回一个char *。但是,返回的指针是指向同一个字符串的指针(不要问我没写它)。
该功能如下所示:
__declspec(dllexport)
char * __cdecl touppercase(char *ps_source)
{
char *ps_buffer = NULL;
assert (ps_source != NULL);
ps_buffer = ps_source;
while (*ps_buffer != '\0')
{
*ps_buffer = toupper(*ps_buffer);
ps_buffer++;
}
*ps_buffer = '\0';
return (ps_source);
}
声明这个的C#代码如下:
[DllImport("mydll.dll", EntryPoint = "touppercase",
CharSet = CharSet.Ansi, ExactSpelling = true,
CallingConvention = CallingConvention.Cdecl)]
private static extern System.IntPtr touppercase(string postData);
我的应用程序中对此的调用类似于
string sTest2 = Marshal.PtrToStringAnsi(to_uppercase(sTest));
然而,sTest2最终只是一个随机字符串。
我使用相同的参数向同一个dll添加了一个测试函数,但这会在本地分配内存并复制字符串。这很好用。为什么原始版本现在有效?
注意:不能自行更新dll库。
答案 0 :(得分:10)
该函数正在修改引用,因此您应该使用stringbuilder:
[DllImport("dll.dll", EntryPoint="touppercase",
CharSet = CharSet.Ansi, ExactSpelling = true,
CallingConvention = CallingConvention.Cdecl)]
private static extern System.IntPtr touppercase(StringBuilder postData);
称之为:
StringBuilder sTest = new StringBuilder("abs");
touppercase(sTest);
string result = sTest.ToString();
关于返回指针的解释 - >本沃格特
答案 1 :(得分:3)
正如您所正确观察的那样,返回类型是指向同一字符串的指针。提供给函数的字符串是.NET用于Unicode-> ANSI转换的临时缓冲区,在P / Invoke返回之前解除分配。所以这是一个必须忽略的狂野指针(最好只使用C#void返回类型)。
我还建议使用StringBuilder
作为参数而不是String
,这让.NET知道该函数将更改传递的参数。 StringBuilder
的内容将由函数修改,即没有返回值的结果。