将一系列接口从C#传递到C ++ / CLI

时间:2010-08-12 05:13:16

标签: c++-cli safearray

我试图将一系列接口从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。)

1 个答案:

答案 0 :(得分:3)

几个问题。首先,您实际上并未在数组中存储VARIANT。这最终不会去任何地方,SafeArray无法存储对托管对象的引用。垃圾收集器移动对象,它无法看到非托管代码所持有的引用,因此无法更新引用。

充其量,您可以创建一个VT_UNKNOWN或VT_DISPATCH数组。但是你无法获得这些托管对象的COM接口指针,它们不是[ComVisible]。当你修复它时,你将使用Marshal.GetIDispatchForObject()或Marshal.GetIUnknownForObject()来获取存储在数组中的接口指针。