关于用Marshal类和IntPtr分配非托管内存的问题(急!若帮忙解决,会加分的)

Frog1228 2011-05-10 01:15:15
项目是关于图像处理的,核心的算法是用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中的一个函数可能没什么问题,但是调用三四次就会报出“尝试读取受保护的内存,这通常指示其它内存已损坏”的错误。
关于这个内存管理问题,已经耗了我一些时间了,若哪位高手能帮忙解决,不胜感激,加分!!!
...全文
1018 10 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
10 条回复
切换为时间正序
请发表友善的回复…
发表回复
Badzpp 2014-01-02
  • 打赏
  • 举报
回复
楼主问题解决了吗?我也在做图像处理,同样遇到了这方面的问题。在VC中输入Int a=1,取其地址&a,转为C# IntPtr类型,此时输入(IntPtr)0没问题,输入(IntPtr)1就出现“尝试读取受保护的内存,这通常指示其它内存已损坏”的错误
Frog1228 2011-05-13
  • 打赏
  • 举报
回复
给分都没有诱惑啊?
gomoku 2011-05-11
  • 打赏
  • 举报
回复
如果是c#方负责分配和释放内存。
你可以传一个byte[]给非托管方。
Frog1228 2011-05-11
  • 打赏
  • 举报
回复
[Quote=引用 4 楼 lizhibin11 的回复:]
2048*1536,2048*1536*3,2048*1536*9这些数组是你的托管程序中没办法直接写入非托管内存,已经定义并且填充数据的double数组对吧?这种情况下如果再用Marshal.Copy等于浪费了同等大小的内存,我觉得没有必要。
可以直接把这些数组作为参数直接传递到API(数组作为参数本身只传送引用),或者如果担心处理时间很长,GC有可能移动这些数组,可以采用GcHandle将这……
[/Quote]
对,是已经已经定义并且填充数据的double数组,用Marshal.Copy,我也觉得没有必要。我怎么会想到用这个的,主要是因为我在一个窗口需要用到另一个窗口中的这些数组的数据,于是我就把这些数组作为参数传递到另一个窗体的构造函数中,在另一个窗体中将数组定义为类的字段,接受传递过来的参数,但是问题就出在将数组定义为类的字段上,程序有时候运行正常,有时候就会在这些类的字段中的数组上报错,说“内存不足”,所以我就用IntPtr代替直接定义数组,后来我看了,其实在类中定义数组的字段,它是引用类型,其实就是和指针一样。
lizhibin11 2011-05-11
  • 打赏
  • 举报
回复
2048*1536,2048*1536*3,2048*1536*9这些数组是你的托管程序中没办法直接写入非托管内存,已经定义并且填充数据的double数组对吧?这种情况下如果再用Marshal.Copy等于浪费了同等大小的内存,我觉得没有必要。
可以直接把这些数组作为参数直接传递到API(数组作为参数本身只传送引用),或者如果担心处理时间很长,GC有可能移动这些数组,可以采用GcHandle将这些数组固定后得到指针,再传递到API,处理完后释放GcHandle。
Frog1228 2011-05-11
  • 打赏
  • 举报
回复
[Quote=引用 4 楼 lizhibin11 的回复:]
2048*1536,2048*1536*3,2048*1536*9这些数组是你的托管程序中没办法直接写入非托管内存,已经定义并且填充数据的double数组对吧?这种情况下如果再用Marshal.Copy等于浪费了同等大小的内存,我觉得没有必要。
可以直接把这些数组作为参数直接传递到API(数组作为参数本身只传送引用),或者如果担心处理时间很长,GC有可能移动这些数组,可以采用GcHandle将这……
[/Quote]
问一个问题,类中的字段是存放在堆中的吧?
Frog1228 2011-05-11
  • 打赏
  • 举报
回复
[Quote=引用 6 楼 gomoku 的回复:]
如果是c#方负责分配和释放内存。
你可以传一个byte[]给非托管方。
[/Quote]
其实不是调用VC写的DLL参数匹配上有问题,除了3是多次调用同一个DLL中的函数会出现错误。其它2个是得到调用DLL的数据进行处理和显示时有问题,具体见5楼我写的问题。
Frog1228 2011-05-10
  • 打赏
  • 举报
回复
啊,怎么没人回答啊?泪奔。。。
Frog1228 2011-05-10
  • 打赏
  • 举报
回复
[Quote=引用 1 楼 bdmh 的回复:]
你这样copy数组肯定不行,这么大的数组,多了,就没有足够的连续空间去容纳它,所以你应该采用分度copy,类似内存映射,每次一小部分,读完了,再都下一部分
[/Quote]
哦,谢谢,如何分度Copy,内存映射怎么弄,有没有什么资料可以提供?
bdmh 2011-05-10
  • 打赏
  • 举报
回复
你这样copy数组肯定不行,这么大的数组,多了,就没有足够的连续空间去容纳它,所以你应该采用分度copy,类似内存映射,每次一小部分,读完了,再都下一部分

111,092

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • AIGC Browser
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧