这有效:
[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,所以我猜内存是通过new
or分配的malloc()
。那现在怎么办,我被圈套了?我的程序中有永久性内存泄漏?
我检查了来源。该字符串是通过创建的static char errmsg[SDL_ERRBUFIZE]
。我的 C 生锈了,但我猜它被声明为static
当它超出函数范围时不会被释放。我不记得静态数组在内存中的位置;有什么方法可以释放它们吗?
编辑:等等......它是静态的。这意味着每次出现新错误时,它都会覆盖旧的错误消息,因此为什么SDL_GetError()
只返回最新的错误消息。因此,我不必担心释放它。
因此,如果所有return: MarshalAs...
选项都试图释放内存,那么唯一的解决方案就是我目前的解决方案。毕竟这是最优的。