C#代码和非托管C ++ lib之间的双向通信

时间:2012-05-29 09:09:38

标签: c# c++ visual-studio-2008 visual-c++ unmanaged

我希望将C ++ lib与来自C#的非托管代码一起使用。我在stackoverflow上发现了一些帖子怎么做。这样它就可以了,但是我需要一些解决方案来用C#代码中的C ++类定义抽象函数(具体来说 - 我的库C ++函数必须在C#中调用函数)。

问候 卡米尔

2 个答案:

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

Calling managed code from unmanaged

Second version of managed<->unmanaged calls

答案 1 :(得分:0)

  

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

这完全没问题,转入函数指针,但是......

  

但是我需要一些解决方案来从C#

中的C ++类定义抽象函数

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

坐下来,学习C ++,C ++中的子类 - 这可以管理C ++(现在是C ++ / CLR),这样你就可以很好地调用托管代码。

但不,你不能将.NET中存储的C ++编译器“.lib”artefact中的类子类化。