我有一个带有以下声明的托管函数(接口和实现):
[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}}数组搞乱了。谢谢!
答案 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无法自动编组本地代码的参数,如果有人可以尝试解释,我仍然会感激。