error C2664: 'innercppclass::getnum' : cannot convert parameter 1 from 'float' to 'float &'
An object from the gc heap (a dereferenced gc pointer) cannot be converted to a native reference
您忘记记录您正在处理的错误,这就是您所看到的。这完全是设计使然,也是托管代码工作方式的基础。
托管函数的 float% 参数可以是指向托管对象的内部指针,例如 ref 类的字段。float& 引用将是运行时的原始非托管指针,指向浮点值。这允许被调用者更新值。两者都只是运行时的普通指针,唯一的区别是垃圾收集器可以看到内部指针,但看不到非托管指针。抖动告诉它在哪里寻找托管指针,对于本机 C++ 函数没有这样的帮助,因为它没有被抖动。
因此,可以为非托管指针分配内部指针值。但是,当垃圾收集器在本机代码运行时运行时,会发生一些非常讨厌的事情。请注意,当程序中的其他线程分配内存时,可能会发生 GC。GC 所做的一件重要事情是压缩堆,它将托管对象作为集合的一部分移动。一个非常理想的特性,它消除了堆中的漏洞,并通过改进引用的局部性使托管代码变得更快。麻烦的是,本机代码持有一个指针,指向该浮动在移动之前所在的位置。如果它通过指针写入,更新浮点值,它将破坏GC 堆。
GC 无法阻止本机代码执行此操作,它不知道指针值在内存中的位置,因此无法更新它。内部指针没有这样的问题,但是对于本地引用来说是一个无法解决的问题。
因此编译器抱怨说,它不可能生成不会使您的程序迟早(通常是稍后)崩溃的代码,并且堆损坏完全无法诊断。您已经找到了解决方法,您需要从不是GC 堆的存储位置生成指针。堆栈很好,局部变量永远不会移动:
void Managed::getnum(float% num) {
float temp;
innercppclass::getnum(temp);
num = temp;
}
否则,当您将 void getnum(float%) 转换为 float getnum() 时,您也会编写这种代码。或者更典型的是在托管代码中,一个属性获取器:
property float num {
float get() {
float temp;
innercppclass::getnum(temp);
return temp;
}
}
您对此无能为力,这是一个非常基本的限制。