我希望将C ++ lib与来自C#的非托管代码一起使用。我在stackoverflow上发现了一些帖子怎么做。这样它就可以了,但是我需要一些解决方案来用C#代码中的C ++类定义抽象函数(具体来说 - 我的库C ++函数必须在C#中调用函数)。
问候 卡米尔
答案 0 :(得分:3)
很酷的问题。为了得到这个想法,你应该考虑以下一般情况。你有一个大的C ++项目,你想用脚本扩展它。在您的情况下,“脚本”是.NET / CLR。
你有:
namespace Native {
class NativeInterface
{
public:
/// Yes, no implementation here, in the "native world"
virtual void NativeMethod() = 0;
/// Some other method(s) you'd like to call
virtual void ConcreteMethod() { cout << "Hello" << endl; }
};
} // namespace Native
您必须为Native :: NativeInfterface创建C ++ / CLI包装器:
namespace Managed
{
ref class NativeInterface
{
public:
virtual void NativeMethod() abstract;
virtual void ConcreteMethod() { NativeObj->ConcreteMethod(); }
public:
Native::NativeInterface* NativeObj;
};
} /// namespace managed
你想用C#:
namespace Managed {
public class MyNETImplementation: NativeInterface
{
override void NativeMethod()
{
DoTheStuff_UsingNativeCode(); // possibly, call the ConcreteMethod
}
}
} // namespace Managed
并且关键是你要将“指针”传递给本地世界某处的MyNETImplementation。
所以你应该在C ++ / CLI中实现棘手的NativeInterfaceImpl类,它将保存对MyNETImplementation的托管引用,并且它将调用适当的方法。
首先尝试:
namespace Managed {
/// C++/CLI in mixed-mode assembly
class NativeInterfaceImpl: public NativeInterface
{
public:
NativeInterfaceImpl(Managed::NativeInterface^ Obj) { ManagedObj = Obj; }
/// Native implementation to call the managed class
virtual void NativeMethod()
{
Obj->NativeMethod();
}
public:
gcroot<Managed::NativeInterface^> ManagedObj;
};
} // namespace Managed
是的,这很棘手,关于如何传递参数有很多问题,但是在这个网站上详细考虑了编组问题。
忘了添加如何使用NativeInterfaceImpl的托管指针。如果使用“gcnew”关键字分配它们,则必须在使用前“固定”它们。
Managed::NativeInterface^ obj = gcnew NativeInterfaceImpl( gcnew MyNETImplementation() );
pin_ptr<NativeInterface> pinned = obj;
/// void SomeNativeFunction(NativeInterface*);
SomeNativeFunction( pinned );
编辑(完成工作):
有一些引用稍微过时(它们与托管C ++有关,而不是C ++ / CLI),但仍然很容易适应。
答案 1 :(得分:0)
具体 - 我的库C ++函数必须在C#中调用函数。
这完全没问题,转入函数指针,但是......
但是我需要一些解决方案来从C#
中的C ++类定义抽象函数
没有办法。您所能做的就是实现与dll的互操作 - 而不是甚至没有标准化的“编译器工件”(在编译器之间有所不同)。
坐下来,学习C ++,C ++中的子类 - 这可以管理C ++(现在是C ++ / CLR),这样你就可以很好地调用托管代码。
但不,你不能将.NET中存储的C ++编译器“.lib”artefact中的类子类化。