我有一个具有 C++ 项目和 C# 项目的解决方案。
C++ 项目定义了一个类,我想在 C# 中实例化它并调用它的成员函数。到目前为止,我设法实例化了这个类:
CFoo Bar = new CFoo();
但是当我尝试调用它的函数时,编译器说,它不可用。
此外,当我在调试器中检查对象时,没有显示任何成员。
我在这里想念什么?
您需要将 C++/CLI 中的类声明为ref class
.
(请注意,我们谈论的是 C++/CLI,而不是 C++。我假设您必须在您的 C++ 项目中启用 CLR,否则您将无法让新CFoo
的工作。)
编辑:
您不需要将所有旧课程转换为ref
课程。
假设你有一些旧的 C++:
class FooUnmanaged
{
int x;
FooUnmanaged() : x(5) {}
};
然后您尝试将其包装在 CLR 类中:
ref class FooManaged
{
FooUnmanaged m;
};
正如您所注意到的,您会收到一条错误消息,指出这是不允许的。但是试试这个:
ref class FooManaged
{
FooUnmanaged *m;
};
这完全没问题。编译器不想分配嵌入在托管堆上的对象中的非托管对象的实例,但是很高兴有一个指针,它会System.IntPtr
在生成的 IL 中变成它。
这意味着您必须决定如何调用delete
. 最可能的解决方案是:
ref class FooManaged
{
FooUnmanaged *u;
public:
FooManaged(FooUnmanaged *u_)
: u(u_) { }
~FooManaged() { delete u; }
};
就像在任何其他 C++ 类中一样。在未来的某个版本中,C++/CLI 可能会自动为我们进行这种翻译。
请注意,生成的 IL 是FooManaged
该类现在实现IDisposable
了 ,并且析构函数已转换为Dispose
方法。这允许 .NET 客户端正确地释放它,例如在 C# 中
using (var m = new FooManaged())
{
// end of block: m will be disposed (and so FooUnmanaged will be deleted)
}