假设这个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
代替吗?答案 0 :(得分:6)
通过互操作边界存储指针是一个非常糟糕的主意,尽一切可能修改C代码来制作字符串的副本。
按照upvoted答案中的建议固定字符串不起作用,pinvoke marshaller必须将字符串转换为char *并在进行本机调用后释放转换后的字符串,使指针无效。你必须自己编组弦乐。将参数声明为IntPtr并使用Marshal.StringToHGlobalAnsi()创建指针值。
或者使用Marshal.StringToCoTaskMemAnsi(),它从COM堆中分配。这是你真正的问题,没有很好的方案来释放字符串。 C代码无法释放字符串,因为它不知道它是如何分配的。您的C#代码无法释放字符串,因为它不知道C代码何时完成指针。这就是复制字符串非常重要的原因。如果您只拨打几次电话,就可以忍受内存泄漏。
答案 1 :(得分:0)
鉴于非托管代码将存储指针而不是复制字符串,我建议您使用GCHandle.Alloc()手动创建字符串的副本并将其作为IntPtr传递。
然后,您将能够管理传递的内存块的生命周期,只有在完成后才能释放它(通过GCHandle)。