我有一个用C ++编写的DLL,它导出一个函数CreateRisk。该函数返回一个接口指针,如下所示:
extern "C"
{
__declspec(dllexport) IRisk* __stdcall CreateRisk()
{
return new Risk();
}
}
IRisk派生自IUnknown,并具有自定义方法Calculate:
class IRisk: public IUnknown
{
public:
virtual int __stdcall Calculate(int i,double s) = 0;
};
类Risk实现IRisk接口(此处省略了实现)。
我想要的是在c#中调用CreateRisk函数并获取对IRisk的引用。
我在c#
中定义了一个包装器接口 [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IRisk
{
[PreserveSig]
int Calculate(int i,double s);
}
我在C#中添加了一个dll条目
[DllImport("Risk.dll")]
extern static IntPtr CreateRisk();
我可以在C#中调用CreateRisk并获取IntPtr的值类型。有没有办法将IntPtr编组到c#IRisk接口,以便我可以在C#中调用Calculate方法?
我曾尝试过Marshal.GetIUnknownForObjectInContext,Marshal.GetObjectForIUnknown,但无济于事。
我知道创建一个COM组件可以做到这一点。但是,COM组件需要在系统中注册。我想避免这些麻烦,让C#直接使用C ++ dll导出的接口。
PS:
以下是我的Risk类实现:
class IRisk : public IUnknown
{
public:
virtual int __stdcall Calculate(int i,double y ) = 0;
};
class Risk:public IRisk
{
int count;
public:
Risk();
virtual int __stdcall Calculate( int i ,double y);
virtual ULONG __stdcall AddRef();
ULONG __stdcall Release();
HRESULT __stdcall QueryInterface(const IID& riid,void **ppvObject);
};
Risk::Risk(){
count=0;
}
ULONG __stdcall Risk::AddRef()
{
return ++count;
}
ULONG __stdcall Risk::Release()
{
return --count;
}
HRESULT __stdcall Risk::QueryInterface(const IID& riid,void **ppvObject) {
*ppvObject=this;
AddRef();
return S_OK;
}
int __stdcall Risk::Calculate(int i ,double y) {
return (int)(i+y);
}
答案 0 :(得分:2)
以下内容可行:
[DllImport("Risk.dll")]
extern static IRiskAssessment CreateRisk();
您需要在界面中添加GUID:
[Guid("your GUID goes here")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IRiskAssessment
{
[PreserveSig]
int Calculate(int i,double s);
}
显然,您需要在C#代码中使用与C ++代码中使用的相同的GUID!
<强>更新强>
查看您的C ++ COM对象实现,明显的错误在QueryInterface
。您不得为所有GUID返回S_OK
。仅为您实现的接口返回S_OK
。其他人返回E_NOINTERFACE
。
答案 1 :(得分:1)
如果我是你,我会使用SWIG为你的C ++代码生成一个C#接口,为你省去很多麻烦。另外,我认为您的程序集可能必须在mixed-mode中编译,或者作为托管程序集编译。