一、描述问题
当托管代码调用非托管代码的时候,经常会出现如下报错:“尝试读取或写入受保护的内存。这通常指示其他内存已损坏”。
二、原因分析
由于非托管代码的内存指针的回收是由非托管代码自身手动完成的,而不是像托管代码一样有统一的垃圾回收机制,比如.NET的GC。
所以对于托管代码的调用方来说无法控制其内存回收。以上问题的产生原因很可能是托管代码调用了已经被回收的非托管对象。封装
的好一点的非托管代码一般都会有内存释放的接口供外部调用,这样调用方就可以管理非托管代码的内存回收。
三、解决方法
尽量不用使用不确定的非托管代码里面提供的对象。如果必须使用的话,实例化完成后直接使用而不要通过什么方法传递,委托回调等方式来
获取里面的对象,在这些复杂未知的过程中,很可能非托管的对象已经被回收了。
四、示例说明
当使用C#调用OpenCVSharp的时候,经常会出现以上问题。
错误代码如下:
1、订阅鼠标回调方法,并将mat的指针Data作为参数传递给回调方法。
1 private void button15_Click(object sender, EventArgs e) 2 { 3 Mat mat = GetMat();//获取Mat对象 4 Cv2.SetMouseCallback("输入图像", GetRGBCvMouseCallback, mat.Data); 5 }
2、回调方法,获取每个点的RGB值
1 private void GetRGBCallbackMethod(MouseEvent @event, int x, int y, MouseEvent flags, IntPtr userdata) 2 { 3 switch (@event) 4 { 5 case MouseEvent.LButtonDown: 6 7 using (Mat mat = new Mat(Rows, Cols, _MatType, userdata))//这种方式会被内存回收,直接在这里面获取对象 8 { 9 for (int i = 0; i < Rows; i++) 10 { 11 for (int j = 0; j < Cols; j++) 12 { 13 Vec3b s = mat.At(j, i);//获取第i0行第i1列) 这个方法会死 尝试读取收保护内存 14 } 15 16 } 17 } 18 break; 19 default: 20 break; 21 }
运行报错,截图如下:
正确代码:
将7行代码Mat mat = new Mat(Rows, Cols, _MatType, userdata)改成using (Mat mat=GetMat()) ,就不会出现以上问题。
原因分析:
原因很可能是userData中回调的过程中已经被回收了,当再次使用时就会报尝试访问被保护的内存的错误。为了
防止此问题的发生,解决办法就是重新实例化一个Mat对象,不使用可能被回收的userData数据。