报错过程:
在C#调试中,遇到报错:
并且提示此结构不为值类。
解决思路:
那么看回代码,代码中用了Marshal.PtrToStructure的函数。是在这一条函数中报的错。当时没有想到是由于传入的参数类型出错。然后打算重写代码,然后偶然发现出错的这个结构体和其他的不一样。于是才怀疑是这个结构的类型有问题,果然,Frame用了struct(结构),而其他能够正常运行的函数都是用了class(类)。
解决方法:
把结构体从struct 改为class。
案例复盘:
虽然问题是成功地解决了,但是原因是什么,到目前为止我还是没有搞懂,到底什么是值类?而为什么PtrToStructure的这个函数又不支持值类型呢?
那首先要学习下什么是值类型及引用类型,
值类型:
值类型(value type):byte,short,int,long,float,double,decimal,char,bool 和 struct 统称为值类型。值类型变量声明后,不管是否已经赋值,编译器为其分配内存。并且数据存放在栈上。
引用类型:
引用类型(reference type):string , class,数组(例如int[]),System.ValueType,System.Enum称为引用类型。当声明一个类时,只在栈中分配一小片内存用于容纳一个地址,而此时并没有为其分配堆上的内存空间。当使用 new 创建一个类的实例时,分配堆上的空间,并把堆上空间的地址保存到栈上分配的小片空间中。
显然地struct属于值类型,因此和报错中的内容是相互排斥的。
第一个问题可以成功解决了,但是第二个问题呢?为什么PtrToStructure这个函数就不能够使用struct作为参数呢?
明明Type就是包括了值类型,而使用这个重载理应是可以的啊。(由于PtrToStructure这个函数包括了两个重载)。然后我观察下他们分别的函数定义,发现原来是有差别的。一个是有返回值的,而另外一个是没有。
所以,我之前一直没有加返回值的话,默认使用的是上面的一个重载,传入的参数为object。
查找C#中 System的命名空间,可以看到Object就是一个class,因此它必须是引用类型而不能是值类型。
于是问题解决了,原来就是因为没有使用具有返回值的重载而没有调用到可以使用值类的那个PtrToStructure。最终这个写法是可以成功的: