5

是的,需要的东西听起来很简单,但事实证明这是一个真正的痛苦。

我有一些C#中的 GUI 代码(注意我以前从未使用过 C#,但熟悉语法)并且我有C++代码,它使用CLI与之交互。

在 C# 中,我想创建一个双精度数组,并将其发送到我的 C++ 代码。我使用下面的代码作为传递数组的方法,并且这是孤立地遵守的。

因此,从 C# 中,我将一个 double[] 数组传递给该函数。

public ref class KernelWrapper
{
public: 
    static void ImageNoiseFilter(System::IntPtr imageData, int imageWidth, int imageHeight, array<double>^ values); 

我应该使用什么参数类型从 C++ 端检索这个数组?

我试过了:

 MyFunction(double values[]){}
 MyFunction(double* values){}
 MyFunction(array<double>^ values){} 

但没有编译,通常最后一个带有“数组不是模板”的消息,并且

Error   1   error C2664: 'RunImageNoiseFilterKernel' : cannot convert parameter 4 from 'cli::array<Type> ^' to 'double *'

任何关于如何实现这一点的提示将不胜感激。


为了可读性,我在这里更新代码

.cpp 文件:

void Bangor::KernelWrapper::ImageNoiseFilter(System::IntPtr imageData, int imageWidth,    int imageHeight, pin_ptr<double> pval){
    RunImageNoiseFilterKernel((Format24bppRgb*)((int)imageData), imageWidth, imageHeight); //If parameter would work, 4th argument would also be passed into this.
}

C#代码:

    double[] randomValues = new double[ARRAY_LENGTH]; //Array of random numbers
    KernelWrapper.ImageNoiseFilter(ptr, image.Width, image.Height, randomValues);

错误是:

Error   1   error C3824: 'cli::pin_ptr<Type>': this type cannot appear in this context (function parameter, return type, or a static member)
Error   3   The best overloaded method match for 'Bangor.KernelWrapper.ImageNoiseFilter(System.IntPtr, int, int, double*)' has some invalid arguments   
Error   4   Argument 4: cannot convert from 'double[]' to 'double*' 

希望这能澄清一点。

4

2 回答 2

3

您应该使用pin_ptr<> 类将托管数组转换为 double*。它固定数组,因此当您的本机函数正在执行和访问数组数据时,垃圾收集器无法移动它。

#include <vcclr.h>                 // Declares cli::pin_ptr<>
using namespace cli;
#pragma managed(push, off)
#  include "unmanagedstuff.h"      // Declaration of MyFunction here
#pragma managed(pop)

...

        pin_ptr<double> pvalues = &values[0];
        MyFunction(pvalues, values->Length);

假设 MyFunction() 肯定也需要知道数组的大小,所以我添加了一个额外的参数来传递数组的大小。不这样做是致命的,您的本机代码可能会破坏托管堆的完整性。请注意,数组只会在pvalues在范围内时保持固定,因此 MyFunction() 不存储传递的指针很重要。如果是这种情况,那么您必须复制数组。

于 2013-03-17T14:51:17.450 回答
2

好的,所以大多数尝试对我来说都失败了;可能是因为我做错了:P,但这是一个可行的解决方案。

进口:

using System.Runtime.InteropServices;
using System.Windows.Forms;
using System.Linq;

在 C# 文件中,我创建了一个数组,如下所示:

    float[] randomValues = new float[ARRAY_LENGTH]; //Array of random numbers
    GCHandle handle = GCHandle.Alloc(randomValues, GCHandleType.Pinned);
    var randPtr = handle.AddrOfPinnedObject();
    KernelWrapper.ImageNoiseFilter(ptr, image.Width, image.Height, randPtr);

ImageNoiseFilter 的定义是:

static void ImageNoiseFilter(System::IntPtr imageData, int imageWidth, int imageHeight, System::IntPtr randValPtr);

为了在需要浮点指针的 C++ 函数中获得,我这样做:

void KernelWrapper::ImageNoiseFilter(System::IntPtr imageData, int imageWidth, int imageHeight, System::IntPtr randValues){
    RunImageNoiseFilterKernel((Format24bppRgb*)((int)imageData), imageWidth, imageHeight, (float*) ((int) randValues));
}

所有这一切似乎都很好:),我不知道这种方法有多好或安全,所以任何阅读本文的人都不应该假设它是“正确”的方法,它只是有效。

于 2013-03-17T20:14:30.720 回答