在P / Invoke中为固定字符串传递什么?

时间:2012-06-11 09:13:52

标签: c# pinvoke pinning

假设这个C函数:

void do_something(const char* str)

它将字符串存储在以后的引用

此外,我在C#中有这个签名来调用这个函数:

[DllImport("NativeLib")]
static extern void do_something(string str);

现在,将字符串传递给此方法时需要做什么:

  • 我是否需要固定字符串(使用GCHandle.Alloc())(或是编组人员创建副本)?
  • 如果我确实需要固定它,我是否会传递“原始”字符串(即我传递给GCHandle.Alloc()的字符串)?或者我是否需要传递GCHandle.AddrOfPinnedObject()的返回值?
  • 在这种情况下,string是否为正确的数据类型(对于do_something)?或者我应该使用IntPtr代替吗?

2 个答案:

答案 0 :(得分:6)

通过互操作边界存储指针是一个非常糟糕的主意,尽一切可能修改C代码来制作字符串的副本。

按照upvoted答案中的建议固定字符串不起作用,pinvoke marshaller必须将字符串转换为char *并在进行本机调用后释放转换后的字符串,使指针无效。你必须自己编组弦乐。将参数声明为IntPtr并使用Marshal.StringToHGlobalAnsi()创建指针值。

或者使用Marshal.StringToCoTaskMemAnsi(),它从COM堆中分配。这是你真正的问题,没有很好的方案来释放字符串。 C代码无法释放字符串,因为它不知道它是如何分配的。您的C#代码无法释放字符串,因为它不知道C代码何时完成指针。这就是复制字符串非常重要的原因。如果您只拨打几次电话,就可以忍受内存泄漏。

答案 1 :(得分:0)

鉴于非托管代码将存储指针而不是复制字符串,我建议您使用GCHandle.Alloc()手动创建字符串的副本并将其作为IntPtr传递。

然后,您将能够管理传递的内存块的生命周期,只有在完成后才能释放它(通过GCHandle)。