我有这个COM方法签名,在C#中声明:
void Next(ref int pcch,[MarshalAs(UnmanagedType.LPArray,SizeParamIndex = 0)] char[] pchText);
我这样称呼:
int cch = 100; var buff = new char[cch]; com.Next(ref cch,buff);
.NET互操作层首先将整个数组复制到临时非托管内存缓冲区,然后将其复制?或者数组是否被自动固定并通过引用传递?
为了尝试,我在COM对象(C)中这样做:
*pcch = 1; pchText[0] = L'A'; pchText[1] = L'\x38F'; // 'Ώ'
当我在返回时检查C#中的buff [1]时,我会得到’Ώ’.但是我不认为这是一个很强的证据,数组被固定,而不是来回复制.
解决方法
我们来做一个小的实验.首先,让我们改变你的COM方法看起来像这样(C):
STDMETHODIMP CComObject::Next(ULONG* pcch,int* addr,OLECHAR* pbuff) { pbuff[0] = L'A'; pbuff[1] = L'\x38F'; *addr = (int)pbuff; *pcch = 1; return S_OK; }
然后,更改C#方法签名:
void Next(ref uint pcch,out IntPtr addr,[In,Out,MarshalAs(UnmanagedType.LPArray,SizeParamIndex = 0)] char[] pbuff);
最后,测试如下:
uint cch = 10; var buff = new char[cch]; IntPtr addr1; unsafe { fixed (char* p = &buff[0]) { addr1 = (IntPtr)p; } } IntPtr addr2; com.Next(ref cch,out addr2,buff); Console.WriteLine(addr1 == addr2);
如预期的那样,addr1 == addr2是true.因此,显然,当传递给COM时,数组确实被固定而不是复制.
也就是说,我找不到任何文档,这将是CLR实现的一个硬要素.例如,Mono可能也可能不是这样.