日期:2014-05-17  浏览次数:21136 次

关于用Marshal类和IntPtr分配非托管内存的问题(急!若帮忙解决,会加分的)
项目是关于图像处理的,核心的算法是用VC写的,我需要用C#调用VC写的DLL。
在工程中我需要定义一些数组字段存放一些数据,因为数组大小比较大,分别是2048*1536,2048*1536*3,2048*1536*9,直接定义,运行的时候就会报错:“内存溢出”,所以我就用IntPtr,结合Marshal类分配非托管的内存,我知道使用Marshal类分配非托管的内存步骤分为以下四个:1. 分配一段非托管的内存空间,用Marshal.AllocHGlobal(int size)  2. 复制数据到这段内存空间中,用Marshal.Copy  3. 把步骤1取得的指针传入非托管函数中或调用函数中  4. 释放这段非托管内存Marshal.FreeHGlobal。
我的疑问是:
1. 用Marshal.AllocHGlobal为数组分配内存,是不是按照数组内容所占的字节数?那double数组2048*1536*3就是一共需要2048*1536*3*8字节大小,即72M字节大小,接着后面连续的三个数组2048*1536分别需要2048*1536*8字节大小,即24M字节大小,共24M*3=72M大小,有时候这段分配内存可以运行过去,但是之后再分配一段2048*1536*3数组的内存就会报出“内存不足”的错误,有时候再一开始分配2048*1536*3数组内存就会报出“内存不足”的错误,我机子的物理内存是2G,虚拟内存是2G-4G,照理说绝对够用了啊,我又没同时运行其他耗内存的东西。
2. 以上是调用DLL中的一个计算比较复杂的函数,其中的算法比较复杂,如果是调用一个计算比较简单地函数,就没有内存不足的错误,但是在另一个窗口得到IntPtr,用Marshal.Copy转化为数组的时候就会报出“尝试读取受保护的内存,这通常指示其它内存已损坏”的错误。
3. 程序中的另一个地方,也要用到IntPtr,但三个数组大小不大,分别是9,2,1,首尾两个是int类型,中间一个是double类型,连续调用一两次DLL中的一个函数可能没什么问题,但是调用三四次就会报出“尝试读取受保护的内存,这通常指示其它内存已损坏”的错误。
关于这个内存管理问题,已经耗了我一些时间了,若哪位高手能帮忙解决,不胜感激,加分!!!

------解决方案--------------------
你这样copy数组肯定不行,这么大的数组,多了,就没有足够的连续空间去容纳它,所以你应该采用分度copy,类似内存映射,每次一小部分,读完了,再都下一部分
------解决方案--------------------
2048*1536,2048*1536*3,2048*1536*9这些数组是你的托管程序中没办法直接写入非托管内存,已经定义并且填充数据的double数组对吧?这种情况下如果再用Marshal.Copy等于浪费了同等大小的内存,我觉得没有必要。
可以直接把这些数组作为参数直接传递到API(数组作为参数本身只传送引用),或者如果担心处理时间很长,GC有可能移动这些数组,可以采用GcHandle将这些数组固定后得到指针,再传递到API,处理完后释放GcHandle。
------解决方案--------------------
如果是c#方负责分配和释放内存。
你可以传一个byte[]给非托管方。