Как преобразовать список массивов в многомерный массив - PullRequest
17 голосов
/ 19 марта 2012

Мне нужно преобразовать следующую коллекцию в double [,]:

 var ret = new List<double[]>();

Все массивы в списке имеют одинаковую длину. Самый простой подход, ret.ToArray(), производит double [] [], что не то, что я хочу. Конечно, я могу создать новый массив вручную и скопировать числа в цикле, но есть ли более элегантный способ?

Редактировать: Моя библиотека вызывается из другого языка, Mathematica, который не был разработан в .Net. Я не думаю, что язык может использовать зубчатые массивы. Я должен вернуть многомерный массив.

Ответы [ 4 ]

24 голосов
/ 19 марта 2012

Я не верю, что для этого есть что-то встроенное в фреймворк - даже Array.Copy в этом случае терпит неудачу. Тем не менее, легко написать код для этого, выполнив цикл:

using System;
using System.Collections.Generic;

class Test
{
    static void Main()
    {
        List<int[]> list = new List<int[]>
        {
            new[] { 1, 2, 3 },
            new[] { 4, 5, 6 },
        };

        int[,] array = CreateRectangularArray(list);
        foreach (int x in array)
        {
            Console.WriteLine(x); // 1, 2, 3, 4, 5, 6
        }
        Console.WriteLine(array[1, 2]); // 6
    }

    static T[,] CreateRectangularArray<T>(IList<T[]> arrays)
    {
        // TODO: Validation and special-casing for arrays.Count == 0
        int minorLength = arrays[0].Length;
        T[,] ret = new T[arrays.Count, minorLength];
        for (int i = 0; i < arrays.Count; i++)
        {
            var array = arrays[i];
            if (array.Length != minorLength)
            {
                throw new ArgumentException
                    ("All arrays must be the same length");
            }
            for (int j = 0; j < minorLength; j++)
            {
                ret[i, j] = array[j];
            }
        }
        return ret;
    }

}
5 голосов
/ 19 марта 2012

Нет простого способа сделать это, потому что в описываемой ситуации нет ничего, что мешало бы массивам double[] в списке иметь разные размеры, что было бы несовместимо с двумерным прямоугольным массивом.Однако, если вы в состоянии гарантировать, что все массивы double[] имеют одинаковую размерность, вы можете создать свой двумерный массив следующим образом:

var arr = new double[ret.Count(),ret[0].Count()];

for( int i=0; i<ret.Count(); i++ ) {
  for( int j=0; j<ret[i].Count(); j++ )
    arr[i,j] = ret[i][j];
}

Это приведет к ошибке времени выполнения, еслилюбой из double[] массивов в списке короче первого, и вы потеряете данные, если какой-либо из массивов будет больше первого.

Если вы действительно решили сохранить массив с неровностямив прямоугольном массиве вы можете использовать «магическое» значение, чтобы указать, что в этой позиции нет значения.Например:

var arr = new double[ret.Count(),ret.Max(x=>x.Count())];

for( int i=0; i<ret.Count(); i++ ) {
  for( int j=0; j<arr.GetLength(1); j++ )
    arr[i,j] = j<ret[i].Count() ? ret[i][j] : Double.NaN;
}

В редакционной заметке я думаю, что это очень плохая идея ™;когда вы собираетесь использовать прямоугольный массив, вы должны постоянно проверять Double.NaN.Кроме того, что если вы хотите использовать Double.NaN в качестве допустимого значения в массиве?Если у вас есть зубчатый массив, вы должны просто оставить его как зубчатый массив.

5 голосов
/ 19 марта 2012

Вы можете сделать следующее как расширение:

    /// <summary>
    /// Conerts source to 2D array.
    /// </summary>
    /// <typeparam name="T">
    /// The type of item that must exist in the source.
    /// </typeparam>
    /// <param name="source">
    /// The source to convert.
    /// </param>
    /// <exception cref="ArgumentNullException">
    /// Thrown if source is null.
    /// </exception>
    /// <returns>
    /// The 2D array of source items.
    /// </returns>
    public static T[,] To2DArray<T>(this IList<IList<T>> source)
    {
        if (source == null)
        {
            throw new ArgumentNullException("source");
        }

        int max = source.Select(l => l).Max(l => l.Count());

        var result = new T[source.Count, max];

        for (int i = 0; i < source.Count; i++)
        {
            for (int j = 0; j < source[i].Count(); j++)
            {
                result[i, j] = source[i][j];
            }
        }

        return result;
    }
4 голосов
/ 19 марта 2012

Если вы собираетесь копировать (я не могу придумать лучшего способа)

var width = ret[0].length;
var length = ret.Count;
var newResult = new double[width, length]
Buffer.BlockCopy(ret.SelectMany(r => r).ToArray(),
                    0, 
                    newResult, 
                    0, 
                    length * width);
return newResult;

РЕДАКТИРОВАТЬ

Я почти уверензацикливание вместо использования SelectMany и ToArray быстрее.

Я знаю, когда меня забросали.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...