如何:使用本机类型声明句柄

不能在本机类型中声明句柄类型。 vcclr.h 提供类型安全的包装器模板 gcroot,以引用 C++ 堆中的 CLR 对象。 此模板允许在本机类型中嵌入虚拟句柄,并将其视为基础类型。 在大多数情况下,可以将 gcroot 对象用作嵌入类型,而无需进行任何强制转换。 但是,对于 for each, in,必须使用 static_cast 来检索基础托管引用。

模板 gcroot 通过值类 System::Runtime::InteropServices::GCHandle 的机构实现,该值类为垃圾回收堆提供“句柄”。 请注意,当 gcroot 类中的析构函数(该函数无法手动调用)不再使用这些句柄时,句柄本身不会被垃圾回收并释放。 如果在本机堆上实例化 gcroot 对象,则必须对该资源调用 delete。

运行时将维护句柄与其引用的 CLR 对象之间的关联性。 当 CLR 对象随垃圾回收堆一起移动时,句柄将返回该对象的新地址。 在将变量分配给 gcroot 模板之前,不必固定变量。

示例

此示例演示如何在本机堆栈上创建 gcroot 对象。

// mcpp_gcroot.cpp
// compile with: /clr
#include <vcclr.h>
using namespace System;

class CppClass {
public:
   gcroot<String^> str;   // can use str as if it were String^
   CppClass() {}
};

int main() {
   CppClass c;
   c.str = gcnew String("hello");
   Console::WriteLine( c.str );   // no cast required
}
hello

此示例演示如何在本机堆上创建 gcroot 对象。

// mcpp_gcroot_2.cpp
// compile with: /clr
// compile with: /clr
#include <vcclr.h>
using namespace System;

struct CppClass {
   gcroot<String ^> * str;
   CppClass() : str(new gcroot<String ^>) {}

   ~CppClass() { delete str; }

};

int main() {
   CppClass c;
   *c.str = gcnew String("hello");
   Console::WriteLine( *c.str );
}
hello

此示例演示如何使用 gcroot 通过对装箱类型使用 gcroot 在本机类型中保留值类型(不是引用类型)的引用。

// mcpp_gcroot_3.cpp
// compile with: /clr
#include < vcclr.h >
using namespace System;

public value struct V {
   String^ str;
};

class Native {
public:
   gcroot< V^ > v_handle;
};

int main() {
   Native native;
   V v;
   native.v_handle = v;
   native.v_handle->str = "Hello";
   Console::WriteLine("String in V: {0}", native.v_handle->str);
}
String in V: Hello

另请参阅

使用 C++ 互操作(隐式 PInvoke)