Инициализация зубчатых массивов

Я хочу создать массив 10 * 10 * 10 в C# like int[][][] (не int[,,]).

Я могу написать код:

int[][][] count = new int[10][][];
for (int i = 0; i < 10; i++) {
    count[i] = new int[10][];
    for (int j = 0; j < 10; j++)
        count[i][j] = new int[10];
}

но я ищу более красивый способ для этого. Может быть что-то вроде этого:

int[][][] count = new int[10][10][10];

8 ответов


int[][][] my3DArray = CreateJaggedArray<int[][][]>(1, 2, 3);

используя

static T CreateJaggedArray<T>(params int[] lengths)
{
    return (T)InitializeJaggedArray(typeof(T).GetElementType(), 0, lengths);
}

static object InitializeJaggedArray(Type type, int index, int[] lengths)
{
    Array array = Array.CreateInstance(type, lengths[index]);
    Type elementType = type.GetElementType();

    if (elementType != null)
    {
        for (int i = 0; i < lengths[index]; i++)
        {
            array.SetValue(
                InitializeJaggedArray(elementType, index + 1, lengths), i);
        }
    }

    return array;
}

вы можете попробовать это:


int[][][] data =
{
    new[]
    {
        new[] {1,2,3}
    }, 
    new[]
    {
        new[] {1,2,3}
    }
};

или без явных значений:


int[][][] data =
{
    new[]
    {
        Enumerable.Range(1, 100).ToArray()
    }, 
    new[]
    {
        Enumerable.Range(2, 100).ToArray()
    }
};


нет встроенного способа создания массива и создания всех элементов в нем, поэтому он не будет даже близко к тому, насколько простым вы хотели бы его видеть. Это будет такая же работа, как и на самом деле.

вы можете создать метод для создания массива и всех объектов в нем:

public static T[] CreateArray<T>(int cnt, Func<T> itemCreator) {
  T[] result = new T[cnt];
  for (int i = 0; i < result.Length; i++) {
    result[i] = itemCreator();
  }
  return result;
}

затем вы можете использовать это для создания трехуровневого зубчатого массива:

int[][][] count = CreateArray<int[][]>(10, () => CreateArray<int[]>(10, () => new int[10]));

трехмерный массив звучит как хороший случай для создания собственного класса. Быть объектно-ориентированным может быть прекрасно.


нет "более элегантного" способа, чем написать 2 for-loops. Вот почему они называются "зубчатыми", размеры каждого поддерева могут отличаться.

но это оставляет вопрос: почему бы не использовать версию [,,]?


int[][][] count = Array.ConvertAll(new bool[10], x =>
                  Array.ConvertAll(new bool[10], y => new int[10]));

вы можете использовать набор данных с одинаковыми таблицами данных. Это может вести себя как 3D-объект (xyz = строка, столбец, таблица)... Но вы собираетесь в конечном итоге с чем-то большим, независимо от того, что вы делаете; вы все еще должны учитывать 1000 пунктов.


С помощью Linq

int[][][] count = new int[10][][].Select(t => new int[10][].Select(tt => new int[10]).ToArray()).ToArray();

это конечно не красиво и, вероятно, не быстро, но это однострочный.