这有效:
[DllImport("SDL2.dll", CallingConvention = CallingConvention.Cdecl, EntryPoint = "SDL_GetError")]
private static extern IntPtr SDL_GetError();
public static string GetError()
{
return Marshal.PtrToStringAnsi(SDL_GetError());
}
这崩溃了:
[DllImport("SDL2.dll", CallingConvention = CallingConvention.Cdecl, EntryPoint = "SDL_GetError")]
[return: MarshalAs(UnmanagedType.LPStr)]
public static extern string GetError();
这篇文章建议 return 属性本质上就像 call 一样Marshal.PtrToStringAnsi,那么有什么关系呢?
正如 Daniel指出的那样,它可能会崩溃,因为编组器正在尝试释放内存。文章还指出,
注意:请注意,非托管端不得使用“new”关键字或“malloc()”C 函数来分配内存。在这些情况下,Interop Marshaler 将无法释放内存。这是因为“new”关键字依赖于编译器,而“malloc”函数依赖于 C 库。
我试过用 , 释放 char 指针Marshal.FreeHGlobal,Marshal.FreeCoTaskMem而且Marshal.FreeBSTR——它们都崩溃了。没有任何其他方法可以释放内存 AFAIK,所以我猜内存是通过newor分配的malloc()。那现在怎么办,我被圈套了?我的程序中有永久性内存泄漏?
我检查了来源。该字符串是通过创建的static char errmsg[SDL_ERRBUFIZE]。我的 C 生锈了,但我猜它被声明为static当它超出函数范围时不会被释放。我不记得静态数组在内存中的位置;有什么方法可以释放它们吗?
编辑:等等......它是静态的。这意味着每次出现新错误时,它都会覆盖旧的错误消息,因此为什么SDL_GetError()只返回最新的错误消息。因此,我不必担心释放它。
因此,如果所有return: MarshalAs...选项都试图释放内存,那么唯一的解决方案就是我目前的解决方案。毕竟这是最优的。