双向通信 C# 代码和非托管C++库
本文关键字:C++ 代码 双向通信 | 更新日期: 2023-09-27 18:35:32
我想C++库与 C# 中的非托管代码一起使用。我在堆栈溢出上发现了一些帖子如何做到这一点。以这种方式它可以工作,但我需要一些解决方案来定义 C# 代码中C++类的抽象函数(具体来说 - 我的函数库C++必须在 C# 中调用函数)。
问候卡米尔
很酷的问题。要获得这个想法,您应该考虑以下一般情况。您有一个大型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
关键是你想把MyNET的"指针"传递给原生世界的某个地方。
因此,您应该在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),但仍然很容易适应。
从非托管调用托管代码
托管<>非托管呼叫的第二版
具体来说 - 我的库C++函数必须在 C# 中调用函数)。
这完全没问题,上交函数指针,但是...
但我需要一些解决方案来定义 C# 中C++类的抽象函数
不可能。您所能做的就是使用 dll 实现互操作 i,而不是使用甚至没有标准化的"编译器工件"(因编译器而异)。
坐下来,学习C++,C++子类 - 这可以C++管理(现在C++/CLR),因此您可以从那里很好地调用托管代码。
但是不可以,您不能在 .NET 中将子类存储在C++编译器".lib"工件中。