Emr*_*ZCU 1 c c# pinvoke dllimport
我正在使用DllImport从我自己的.net类中调用c包装器库中的方法.c dll中的此方法创建一个字符串变量并返回该字符串的指针.
像这样的东西;
_declspec(dllexport) int ReturnString()
{
char* retval = (char *) malloc(125);
strcat(retval, "SOMETEXT");
strcat(retval, "SOMETEXT MORE");
return (int)retval;
}
Run Code Online (Sandbox Code Playgroud)
然后我使用Marshall.PtrToStringAnsi(ptr)读取字符串.在我得到字符串的副本后,我只需调用另一个c方法HeapDestroy,该方法位于调用free(ptr)的c包装器库中.
这是个问题; 最近虽然它像魅力一样工作,但我开始"尝试读取或写入受保护的内存区域"例外.经过深入分析,我想通了,我相信,虽然我为这个指针调用了自由方法,指针的值没有被清除,并且这填充了无人看管的堆并使我的iis工作进程抛出此异常.顺便说一句,它是一个在c库中调用此方法的网站项目.
你能帮我解决这个问题吗?
当然,这是C#代码;
[DllImport("MyCWrapper.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
private extern static int ReturnString();
[DllImport("MyCWrapper.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
private extern static void HeapDestroy(int ptr);
public static string GetString()
{
try
{
int i = ReturnString();
string result = String.Empty;
if (i > 0)
{
IntPtr ptr = new IntPtr(i);
result = Marshal.PtrToStringAnsi(ptr);
HeapDestroy(i);
}
return result;
}
catch (Exception e)
{
return String.Empty;
}
}
Run Code Online (Sandbox Code Playgroud)
可能是底层C代码的问题.您没有向strcat依赖的字符串添加NULL终止符(或检查malloc的NULL返回).在这种情况下很容易损坏内存.您可以通过执行以下操作来解决此问题.
retval[0] = '\0';
strcat(retval, "SOMETEXT");
Run Code Online (Sandbox Code Playgroud)
问题的另一部分是你在系统上玩弄技巧.最好正确编写并让系统正常运行代码.第一步是修复本机代码以正确返回字符串.您需要考虑的一件事是CLR(HGlobal和CoTask分配)本身可以释放某些类型的内存.所以让我们改变函数签名返回char*一个不同的分配器.
_declspec(dllexport) char* ReturnString()
{
char* retval = (char *) CoTaskMemAlloc(125);
retval[0] = '\0';
strcat(retval, "SOMETEXT");
strcat(retval, "SOMETEXT MORE");
return retval;
}
Run Code Online (Sandbox Code Playgroud)
然后,您可以使用以下C#签名并将IntPtr与Marshal.FreeCoTaskMem一起释放.
[DllImport("SomeDll.dll")]
public static extern IntPtr ReturnString();
Run Code Online (Sandbox Code Playgroud)
虽然更好.当编组时,如果CLR认为它需要释放内存,它将使用FreeCoTaskMem来执行此操作.这通常与字符串返回相关.由于您使用CoTaskMemAlloc分配了内存,因此您可以自行保存编组+释放步骤并执行以下操作
[DllImport("SomeDll.dll", CharSet=Ansi)]
public static extern String ReturnString();
Run Code Online (Sandbox Code Playgroud)