2

我以前从不需要在 C# 中使用指针,但是,我使用的库要求方法参数作为指针传递。该库允许使用 SIMD 指令集。

为了测试如何使用该库,我尝试编写一种使用 SIMD 一次性计算数组所有元素的余弦值的方法。

这就是我所拥有的:

double[] ValuesToCalculate = new double[MAX_SIZE];
double[] CalculatedCosines = new double[MAX_SIZE];

long Result;
Result = CalculateCosineArray(ValuesToCalculate, CalculatedCosines);

public static long CalculateCosineArraySIMD(double[] array, double[] result)
{
     Stopwatch stopwatch = new Stopwatch();
     stopwatch.Start();
     for (int i = 0; i < array.Length; i++)
     {
         Yeppp.Math.Cos_V64f_V64f(*array, result, MAX_SIZE);
     }

     stopwatch.Stop();
     return stopwatch.ElapsedMilliseconds;
}

但是,我收到这些错误:

The best overloaded method match for 'Yeppp.Math.Cos_V64f_V64f(double*, double*, int)' has some invalid arguments
Argument 1: cannot convert from 'double[]' to 'double*'
The * or -> operator must be applied to a pointer
Argument 2: cannot convert from 'double[]' to 'double*'

如何获得在此代码中工作的指针?同样,这是我在使用 C# 时第一次出现指针。

4

2 回答 2

5

您必须使用fixed语句从数组中获取指针。

大概是这样的:

public static long CalculateCosineArraySIMD(double[] array, double[] result)
{
    Stopwatch stopwatch = new Stopwatch();
    stopwatch.Start();

    fixed (double* arrayPtr = array, resultPtr = result)
    {
        Yeppp.Math.Cos_V64f_V64f(arrayPtr, resultPtr, MAX_SIZE);
    };

    stopwatch.Stop();
    return stopwatch.ElapsedMilliseconds;
}
于 2014-02-06T01:41:45.560 回答
4

Yeppp 中的方法!库有两个重载:一个采用数组 + 偏移量,另一个采用指针(例如,如果您想使用stackalloc的内存调用它们)。例如,用于余弦计算 Yeppp!提供两个重载:

因此,您的示例应重写为:

double[] ValuesToCalculate = new double[MAX_SIZE];
double[] CalculatedCosines = new double[MAX_SIZE];

long Result;
Result = CalculateCosineArray(ValuesToCalculate, CalculatedCosines);

public static long CalculateCosineArraySIMD(double[] array, double[] result)
{
    Stopwatch stopwatch = new Stopwatch();
    stopwatch.Start();

    Yeppp.Math.Cos_V64f_V64f(array, 0, result, 0, MAX_SIZE);

    stopwatch.Stop();
    return stopwatch.ElapsedMilliseconds;
}

请注意,每个数组只需要一次调用(即使您使用了指针)。

于 2014-02-06T02:46:39.933 回答