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

Seb*_*ski 5 c# pinvoke pinning

假设这个C函数:

void do_something(const char* str)
Run Code Online (Sandbox Code Playgroud)

它将字符串存储在某处以供以后参考.

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

[DllImport("NativeLib")]
static extern void do_something(string str);
Run Code Online (Sandbox Code Playgroud)

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

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

Han*_*ant 7

通过互操作边界存储指针是一个非常糟糕的想法,尽你所能修改C代码来制作字符串的副本.

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

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