3

我有以下 C++ 方法:

__declspec(dllexport) void __stdcall getDoubles(int *count, double **values); 该方法分配并填充一个 double 数组并将 *count 设置为数组的大小。

我设法通过pinvoke让它工作的唯一方法是:

[System.Runtime.InteropServices.DllImportAttribute("xx.dll")]
 public static extern void getDoubles(ref int count, ref System.IntPtr values);

用法是:

int count = 0;
IntPtr doubles = new IntPtr();
Nappy.getDoubles(ref count, ref doubles);
double[] dvs = new double[count];
for(int i = 0;i < count;++{
    dvs[i] = (double)Marshal.PtrToStructure(doubles, typeof(System.Double));
    doubles = new IntPtr(doubles.ToInt64()+Marshal.SizeOf(typeof(System.Double)));
}

这些值最终出现在 dvs 数组中。有没有更好的方法来做到这一点而不涉及托管语言中的指针算术......

4

2 回答 2

3

我认为你可以使用

Marshal.Copy( source, destination, 0, size );
于 2009-01-14T12:32:38.163 回答
1

您需要更改非托管方法签名,使其如下所示:

__declspec(dllexport) void __stdcall getDoubles(SAFEARRAY *array); 

然后您应该能够使用该功能的以下托管版本:

[System.Runtime.InteropServices.DllImportAttribute("xx.dll")]
public static extern void getDoubles(
              [MarshalAs(UnmanagedType.SafeArray, SafeArraySubType=VT_R8)]
              double[] array
);

当然,您还必须重写非托管代码才能使用 SAFEARRAY。可以在 MSDN 上找到有关此主题的更多信息。

但是有一个问题,我记得在 C# 中使用 ZLib,它能够在没有任何包装器的情况下使用 byte[] 而非托管对应物是 BYTE*,您是否尝试过直接使用 double* / double[] ?

于 2009-01-14T12:56:21.873 回答