2

我想将 C++ lib 与来自 C# 的非托管代码一起使用。我在stackoverflow上发现了一些帖子是如何做到的。通过这种方式它可以工作,但我需要一些解决方案来在 C# 代码中从 C++ 类定义抽象函数(具体来说 - 我的库 C++ 函数必须在 C# 中调用函数)。

问候卡米尔

4

2 回答 2

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),但仍然很容易适应。

从非托管调用托管代码

托管<->非托管调用的第二个版本

于 2012-05-29T09:28:14.897 回答
0

具体来说 - 我的库 C++ 函数必须在 C# 中调用函数)。

这完全没问题,提交函数指针,但是......

但我需要一些解决方案来从 C# 中的 C++ 类定义抽象函数

没门。您所能做的就是实现与 dll 的互操作 - 而不是使用甚至没有标准化的“编译器工件”(编译器之间有所不同)。

坐下来,学习 C++,C++ 中的子类——这可以被 C++ 管理(现在是 C++/CLR),所以你可以从那里很好地调用托管代码。

但是不,您不能子类化存储在 .NET 中的 C++ 编译器“.lib”工件中的类。

于 2012-05-29T09:27:04.337 回答