在 C# 中常规访问多维数组

本文关键字:数组 访问 常规 | 更新日期: 2023-09-27 18:31:32

C# 允许创建和填充多维数组,下面是一个简单的示例:

    public static void Main(String[] args)
    {
        var arr = (int[,])CreateArray(new [] {2, 3}, 8);
        Console.WriteLine("Value: " + arr[0,0]);
    }
    // Creates a multidimensional array with the given dimensions, and assigns the
    // given x to the first array element
    public static Array CreateArray<T>(int[] dimLengths, T x)
    {
        var arr = Array.CreateInstance(typeof(T), dimLengths);
        var indices = new int[dimLengths.Length];
        for (var i = 0; i < indices.Length; i++)
            indices[i] = 0;
        arr.SetValue(x, indices);  // Does boxing/unboxing
        return arr;
    }

这很好用。但是,由于某种原因,没有 Array.SetValue() 的通用版本,所以上面的代码会进行装箱/取消装箱,我想避免这种情况。我想知道我是否错过了什么,或者这是否是 .NET API 中的遗漏?

在 C# 中常规访问多维数组

不,您没有遗漏任何内容:Arrays没有设置值而不装箱和取消装箱的选项。

使用 LINQ 确实有另一种选择,但它可能比单个元素的装箱/取消装箱慢,因为编译动态 lambda 会"吃掉"潜在的好处:

public static Array CreateArray<T>(int[] dimLengths, T x) {
    var arr = Array.CreateInstance(typeof(T), dimLengths);
    var p = Expression.Parameter(typeof(object), "arr");
    var ind = new Expression[dimLengths.Length];
    for (var i = 0; i < dimLengths.Length; i++) {
        ind[i] = Expression.Constant(0);
    }
    var v = Expression.Variable(arr.GetType(), "cast");
    var block = Expression.Block(
        new[] {v}
    ,   new Expression[] {
            Expression.Assign(v, Expression.Convert(p, arr.GetType()))
        ,   Expression.Assign(Expression.ArrayAccess(v, ind), Expression.Constant(x))
        ,   Expression.Constant(null, typeof(object))
        }
    );
    Expression.Lambda<Func<object, object>>(block, p).Compile()(arr);
    return arr;
}

如果要在一个循环中设置所有元素,可以修改上述内容以编译具有多个嵌套循环的动态创建的 lambda。在这种情况下,您可以改进必须在一系列嵌套循环中执行多个装箱和取消装箱的情况。

由于某种原因,没有通用版本的Array.SetValue()

虽然绝对可以编写类似于Array类中的SetValue的泛型方法,但可能不希望这样做。非泛型类上的泛型方法会给出编译时类型安全性的错误承诺,这是无法保证的,因为编译器不知道Array对象的运行时类型。

我也没有找到任何通用方法来将值设置到 Array 实例中,所以我想唯一的解决方法是使用不安全的上下文来避免装箱。

但是,现在我想起来就没有通用版本了。请参阅,当您定义泛型方法method<T>()...时,您确实定义了该方法的参数:...<T>(T[] a)...您必须具体说明维度计数,这是一个。要创建二维参数,请像这样定义它...<T>(T[,] a)...依此类推。

如您所见,根据 C# 的当前语法,您根本无法创建可以接受任何维数组的泛型方法。