编组从托管代码到本机代码的字符串数组

时间:2012-11-30 23:14:26

标签: c# c++ .net interop marshalling

我有一个带有以下声明的托管函数(接口和实现):

[return: MarshalAs(UnmanagedType.SafeArray, SafeArraySubType = VarEnum.VT_BSTR)]
String[] ManagedFunction()
{
    String[] foo = new String[1];
    foo[0] = "bar";
    return foo;
}

还有一个本机C ++接口,其具有与托管接口相同的方法,在该接口内部,此方法具有以下声明:

void ManagedFunction(SAFEARRAY* foo); 

本机代码通过以下方式调用此函数:

void NativeFunction(ManagedBinding binding)
{
    CComSafeArray<BSTR> cComSafeArray;
    cComSafeArray.Create(); 
    LPSAFEARRAY safeArray = cComSafeArray.Detach();
    binding.comObject->ManagedFunction(safeArray); 
}

我不确定我做错了什么但是在调用托管函数后,safeArray似乎有垃圾值,在将返回值编组回本机代码时出现问题。有没有比.Net interop更多经验的人能否对此有所了解?另外,可能有必要提一下,如果您对我的托管函数ValueType返回boolean有疑问,我可能会遇到问题。 1}}数组搞乱了。谢谢!

2 个答案:

答案 0 :(得分:0)

1)你的函数返回SAFEARRAY,为什么在调用函数之前分配它?
2)ManagedFunction应该返回SAFEARRAY,所以它应该得到一个SAFEARRAY*才能返回它!所以你应该说:

LPSAFEARRAY lpsa;
binding.comObject->ManagedFunction(&lpsa);
CComSafeArray<BSTR> cComSafeArray;
cComSafeArray.Attach(lpsa);

答案 1 :(得分:0)

好吧,我终于开始工作了。我创建了一个名为SAFEARRAY ManagedSafeArray的托管代表(从此处被盗:http://social.msdn.microsoft.com/Forums/en-US/clr/thread/6641abfc-3a9c-4976-a523-43890b2b79a2/):

[StructLayout(LayoutKind.Sequential)]
struct ManagedSafeArray
{
    public ushort   dimensions;     // Count of dimensions in the SAFEARRAY
    public ushort   features;       // Flags to describe SAFEARRAY usage
    public uint     elementSize;    // Size of an array element
    public uint     locks;          // Number of times locked without unlocking
    public IntPtr   dataPtr;        // Pointer to the array data
    public uint     elementCount;   // Element count for first (only) dimension
    public int      lowerBound;     // Lower bound for first (only) dimension
}

我将方法的签名更改为:

void ManagedMethod(ref ManagedSafeArray foo);

在我的方法中,我通过调用dataPtr手动更新了Marshal.AllocCoTaskMem(...)字段,然后复制了我希望SAFEARRAY包含的字符串。

我不知道为什么CLR无法自动编组本地代码的参数,如果有人可以尝试解释,我仍然会感激。