我试图将一系列接口从C#传递给C ++ / CLI。这是代码:
// *** SafeArrayTesting_PlusPlus.cpp ***
#include "stdafx.h"
#include <comdef.h>
using namespace System;
using namespace System::Runtime::InteropServices;
namespace SafeArrayTesting_PlusPlus {
public ref class MyCppClass
{
public:
MyCppClass();
~MyCppClass();
void SetMyInterfaces(
array<SafeArrayTesting_Sharp::MyInterface^>^ myInterfaces);
};
MyCppClass::MyCppClass(){}
MyCppClass::~MyCppClass(){}
void MyCppClass::SetMyInterfaces(array<SafeArrayTesting_Sharp::MyInterface^>^
myInterfaces)
{
// Create safearray
SAFEARRAY *safeArrayPointer;
SAFEARRAYBOUND arrayDim[1]; // one dimensional array
arrayDim[0].lLbound= 0;
arrayDim[0].cElements= myInterfaces->Length;
safeArrayPointer = SafeArrayCreate(VT_UNKNOWN,1,arrayDim);
// copy ints to safearray
for (long lo= 0;lo<myInterfaces->Length;lo++)
{
IntPtr myIntPtr = Marshal::GetIUnknkownForObject(myInterfaces[lo]);
SafeArrayPutElement(
safeArrayPointer,
&lo,
static_cast<void*>(myIntPtr)
);
}
// do something with the safearray here - area XX
}}
// *** SafeArrayTesting_Main.cs ***
using SafeArrayTesting_PlusPlus;
using SafeArrayTesting_Sharp;
namespace SafeArrayTesting_Main
{
class SafeArrayTesting_Main
{
static void Main()
{
var myCppClass = new MyCppClass();
MyInterface myInterface = new MyClass();
myCppClass.SetMyInterfaces(new[]{ myInterface });
}
}}
// *** SafeArrayTesting_Sharp.cs ***
using System;
using System.Runtime.InteropServices;
namespace SafeArrayTesting_Sharp
{
[ComVisible(true)]
public interface MyInterface
{
int MyInt { get; set; }
string MyString { get; set; }
DateTime MyDateTime { get; set; }
}
[ComVisible(true)]
public class MyClass : MyInterface
{
public int MyInt{get;set;}
public string MyString{get;set;}
public DateTime MyDateTime{get; set;}
}
// Just to please the compiler; bear with me.
class DummyClass { static void Main() { } }
}
正如这里所写,代码运行和编译干净。但是,当运行“区域XX”部分时,我得到System.Runtime.InteropServices.SEHException
。
XX代码只是一行,它调用自动生成的方法接受SAFEARRAY指针。以下是此方法的声明(来自.tlh文件):
virtual HRESULT __stdcall put_SafeArray (
/*[in]*/ SAFEARRAY * pRetVal ) = 0;
我实际上认为这种方法将SAFEARRAY转换回.NET数组 - 它是我公司当时运行的转换项目的一部分。所以除了使用SAFEARRAY之外别无选择。
无论如何,如果没有XX部分的代码没有错误,那真的会让我感到惊讶;在C ++方面,我是一个新手。你能帮我发现一些问题吗?如果有人能提出更好的方法来测试SAFEARRAY的有效性,那也是一种帮助。
(顺便说一下,这是问题SafeArrayPutElement method throws System.AccessViolationException的一个更复杂的变体,其中我只是将一组int从C#传递给C ++ / CLI。)
答案 0 :(得分:3)
几个问题。首先,您实际上并未在数组中存储VARIANT。这最终不会去任何地方,SafeArray无法存储对托管对象的引用。垃圾收集器移动对象,它无法看到非托管代码所持有的引用,因此无法更新引用。
充其量,您可以创建一个VT_UNKNOWN或VT_DISPATCH数组。但是你无法获得这些托管对象的COM接口指针,它们不是[ComVisible]。当你修复它时,你将使用Marshal.GetIDispatchForObject()或Marshal.GetIUnknownForObject()来获取存储在数组中的接口指针。