c# - 从 c# COM dll 返回时,C++ SAFEARRAY 具有无效数据

标签 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 我收到此警告:“类型库导出器警告处理'[...]。下载,[...]。警告:引用类型具有顺序或显式布局,因此作为结构导出。”但我想它不应该导致完全无效的返回数据

打印结果如下:

 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

但是每次运行它们都会改变一点。

所以我希望任何人都可以帮助我找到我错过的细节^^ 感谢阅读并随时询问更多详细信息

最佳答案

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

SAFEARRAY *psa = pDB->GetData();
if (psa)
{
    SafeArrayLock(psa);
    LONG lb;
    LONG ub;
    SafeArrayGetLBound(psa, 1, &lb);
    SafeArrayGetUBound(psa, 1, &ub);
    for (int i = lb; i <= ub; i++)
    {
        _DownloadPtr download(((IUnknown**)psa->pvData)[i]);
        wprintf(L"%s\n", (LPWSTR)download->GetTarget());
    }
    SafeArrayUnlock(psa);
}

请注意,这意味着您的 Download 类也必须标记为 [ComVisible(true)],并将字段转换为公共(public)属性。因为在这个例子中你需要一个 IUnknown 接口(interface)(为了看到像 GetTarget() 这样的方法自动添加到生成的 C++ 代码中),我还建议你添加一个 [ClassInterface(ClassInterfaceType.AutoDual)] 属性到

如果您确实需要 VARIANT 数组(为什么要这样做?),那么您将不得不使用 object[] 而不是类型化的 objects.arrays。

关于c# - 从 c# COM dll 返回时,C++ SAFEARRAY 具有无效数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45754483/

相关文章:

c# - 如何在 selenium 中获取表行 ID

c++ - 如何在图中找到 3 条边的负加权循环?

c++ - 从与静态运行时(/MT 或/MTd)链接的 DLL 函数返回非原始 C++ 类型

javascript - Javascript 的 Object.assign() 在 C# 中的等价物是什么

c# - 服务安装程序类可以在不同的程序集中运行吗?

c# - 在C#中,类或函数前方括号中写的是什么?

c++ - fstream C++ 的相对路径

c++ - 可以在不附加到帧缓冲区的情况下复制渲染缓冲区吗?

c - DLL 导出 __stdcall 不带 DEF 文件并用 VS 修饰函数名称

c# - 如何卸载已加载的程序集