C++ 安全数组在从 c# COM dll 返回时具有无效数据

C++ SAFEARRAY has invalid data when returned from c# COM dll

本文关键字:返回 数据 无效 dll COM 数组 安全 C++      更新时间:2023-10-16

我尝试从C ++中的C#DLL中获取数据(以类/结构数组的形式(。 我尝试使用安全数组帮助程序函数,但我收到的所有数据都是无效的......

在调试一切正常/看起来很完美时,直到我到达接收 DLL 中生成的数据的点(只有 SAFEARRAY* 看起来无效(,所以可能是C++应用程序和 C# COM DLL 之间的通信存在问题 VS 安全阵列调试/自动窗口 VS 下载列表调试/自动窗口

下面是一个代码示例: 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;
}

但相同的结果。 对于元帅,我收到此警告:"类型库导出器警告处理'[...]。下载,[...]'。警告:引用类型具有顺序或显式布局,因此导出为结构。 但我想它不应该导致完全无效的返回数据

打印结果如下所示:

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"%sn", (LPWSTR)download->GetTarget());
}
SafeArrayUnlock(psa);
}

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

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