双向通信 C# 代码和非托管C++库

本文关键字:C++ 代码 双向通信 | 更新日期: 2023-09-27 18:35:32

我想C++库与 C# 中的非托管代码一起使用。我在堆栈溢出上发现了一些帖子如何做到这一点。以这种方式它可以工作,但我需要一些解决方案来定义 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"工件中。