从c#COM dll

时间:2017-08-18 10:30:56

标签: c# c++ dll com

我尝试从c ++中的c#dll获取数据(以类/结构数组的形式)。我尝试使用safearray辅助函数,但我收到的所有数据都是无效的....

虽然调试一切工作/看起来很完美,直到我达到重新生成在dll中生成的数据的点(只有SAFEARRAY *看起来无效),所以可能是C ++应用程序和c#COM dll之间的通信问题 VS SAFEARRAY Debug/Auto Window VS DownloadList Debug/Auto Window

这是一个代码示例: C#

// data class
public class Download
{
    public string Target;
    public string Data;
    public int Port;
}

// function called from outside dll
public Download[] GetData()
{
    [...]
    return DownloadList.ToArray(); // List<Download>
}

C ++

#import "[...]/MSSQL_Lib.tlb"
// pDB is a class instance which contain the GetData func
[...]
SAFEARRAY* Data = pDB->GetData();
if( Data != nullptr )
{
    // print varian type info from result
    SafeArrayLock( Data );

    VARIANT* ValueArray = (VARIANT*)Data->pvData;
    long Lower = 0, Upper = 0;
    SafeArrayGetLBound( Data, 1, &Lower );
    SafeArrayGetUBound( Data, 1, &Upper );

    for( long i = 0; i <= ( Upper - Lower ); ++i )
    {
        PrintVariant( &ValueArray[i] );
    }

    SafeArrayUnlock( Data );
    SafeArrayDestroy( Data );
}
[...]
// function end

void PrintVariant( VARIANT* pV )
{
    switch( pV->vt )
    {
        case VT_BSTR:
            wprintf( L" String : %s \n", pV->bstrVal );
            break;
        default:
            wprintf( L" Unrecognized Type : %d \n", pV->vt );
            break;
    }
}

我还试图编组c#类:

[StructLayout(LayoutKind.Sequential)]
public class Download
{
    [MarshalAs(UnmanagedType.BStr)]
    public string Target;
    [MarshalAs(UnmanagedType.BStr)]
    public string Data;
    public int Port;
}

但结果相同。与marshal我收到此警告:&#34;键入库导出器警告处理&#39; [...]。下载,[...]&#39;。警告:引用类型具有顺序或显式布局,因此导出为结构。&#34;但我想它不应该导致完全无效的返回数据

打印结果如下:

 Unrecognized Type : 65432
 Unrecognized Type : 65048
 Unrecognized Type : 64664
 Unrecognized Type : 64280
 Unrecognized Type : 1
 Unrecognized Type : 0
 Unrecognized Type : 0
 Unrecognized Type : 43008
 Unrecognized Type : 21288
 Unrecognized Type : 1331

但他们会改变每一次运行。

所以我希望有人能帮助我,找到我错过的细节^^ 阅读时请随时询问更多详情

1 个答案:

答案 0 :(得分:0)

您的数组不是VARIANT数组,而是IUnknown *数组(对SafeArrayGetVartype的调用会告诉您)。您必须更改以下内容的代码:

[ComVisible(true)]

请注意,这意味着您的下载类也必须标记为[ClassInterface(ClassInterfaceType.AutoDual)],并将字段转换为公共属性。因为在此示例中需要一个IUnknown接口(要查看自动添加到生成的C ++代码中的GetTarget()等方法),我还建议您为其添加object[]属性。

如果您确实需要VARIANT数组(为什么会这样?),那么您必须使用worksheets("Destination").range("A1").value=worksheets("Source").range("a1").value而不是类型化objects.arrays。