Каков наилучший способ извлечь одномерный массив из прямоугольного массива в C #?

StackOverflow https://stackoverflow.com/questions/232545

  •  04-07-2019
  •  | 
  •  

Вопрос

Скажем, у меня есть массив прямоугольных строк, а не неровный массив

string[,] strings = new string[8, 3];

Каков наилучший способ извлечь из этого одномерный массив (одну строку или один столбец)? Конечно, я могу сделать это с помощью цикла for, но я надеюсь, что в .NET встроен более элегантный способ.

Бонусные баллы за преобразование извлеченного массива строк в массив объектов.

Это было полезно?

Решение

Для прямоугольного массива:

string[,] rectArray = new string[3,3] { 
    {"a", "b", "c"}, 
    {"d", "e", "f"}, 
    {"g", "h", "i"} };

var rectResult = rectArray.Cast<object>().ToArray();

И для неровного массива:

string[][] jaggedArray =  { 
    new string[] {"a", "b", "c", "d"}, 
    new string[] {"e", "f"}, 
    new string[] {"g", "h", "i"} };

var jaggedResult = jaggedArray.SelectMany(s => s).Cast<object>().ToArray();

Другие советы

Вы можете тривиально преобразовать массив строк в массив объектов - другой путь не работает. Фактическое извлечение имеет для использования цикла for, насколько я вижу: Array.Copy требует, чтобы ранги источника и цели были одинаковыми, а Buffer.BlockCopy работает только для массивов типов значений. Хотя это кажется странным ...

Вы можете использовать LINQ для добавления строки или столбца в одном выражении, хотя это будет неэффективно (так как он создаст список внутренне, затем придется преобразовать его в массив - если вы сделаете это самостоятельно, вы можете предварительно выделить массив до нужного размера и скопировать напрямую).

Копирование строки ( rowNum - это строка, которую нужно скопировать):

object[] row = Enumerable.Range(0, rowLength)
                         .Select(colNum => (object) stringArray[rowNum, colNum])
                         .ToArray();

Копирование столбца ( colNum - это столбец, который нужно скопировать):

object[] column = Enumerable.Range(0, columnLength)
                            .Select(rowNum => (object) stringArray[rowNum, colNum])
                            .ToArray();

Я не уверен, что это действительно лучше / проще, чем цикл foreach - особенно если вы пишете метод ExtractRow и метод ExtractColumn и используете их повторно .

Я просто хотел бы уточнить (учитывая пример и то, что спрашивают).

зубчатый массив - это массив массивов, который объявлен так:

string[][] data = new string[3][];
data[0] = new string[] { "0,[0]", "0,[1]", "0,[2]" };
data[1] = new string[] { "1,[0]", "1,[1]", "1,[2]" ];
data[2] = new string[] { "2,[0]", "1,[1]", "1,[2]" };

По сравнению с прямоугольным массивом , определенным как один массив, который содержит несколько измерений:

string[,] data = new string[3,3];
data[0,0] = "0,0";
data[0,1] = "0,1";
data[0,2] = "0,2";
...etc

Из-за этого зубчатый массив является IQueryable / IEnumerable, поскольку вы можете выполнять итерацию по нему для получения массива на каждой итерации. Принимая во внимание, что прямоугольный массив является не IQueryable / IEnumerable, так как элементы адресуются в полном измерении (0,0 0,1..etc), поэтому вы в этом случае не будет возможности использовать Linq или любые предопределенные функции, созданные для Array.

Хотя вы можете итерировать массив один раз (и достичь желаемого) следующим образом:

/// INPUT: rowIndex, OUTPUT: An object[] of data for that row
int colLength = stringArray.GetLength(1);
object[] rowData = new object[colLength];
for (int col = 0; col < colLength; col++) {
    rowData[col] = stringArray[rowIndex, col] as object;
}
return rowData;

/// INPUT: colIndex, OUTPUT: An object[] of data for that column
int rowLength = stringArray.GetLength(0);
object[] colData = new object[rowLength];
for (int row = 0; r < rowLength; row++) {
    colData[row] = stringArray[row, colIndex] as object;
}
return colData;

Надеюсь, это поможет:)

LINQ - это ответ

static object[] GetColumn(string[][] source, int col) {
    return source.Iterate().Select(x => source[x.Index][col]).Cast<object>().ToArray();
}
static object[] GetRow(string[][] source, int row) {
    return source.Skip(row).First().Cast<object>().ToArray();
}
public class Pair<T> {
    public int Index;
    public T Value;
    public Pair(int i, T v) {
        Index = i;
        Value = v;
    }
}
static IEnumerable<Pair<T>> Iterate<T>(this IEnumerable<T> source) {
    int index = 0;
    foreach (var cur in source) {
        yield return new Pair<T>(index, cur);
        index++;
    }
}

Строки можно легко скопировать с помощью Array.Copy:

        int[][] arDouble = new int[2][];
        arDouble[0] = new int[2];
        arDouble[1] = new int[2];
        arDouble[0][0] = 1;
        arDouble[0][1] = 2;
        arDouble[1][0] = 3;
        arDouble[1][1] = 4;

        int[] arSingle = new int[arDouble[0].Length];

        Array.Copy(arDouble[0], arSingle, arDouble[0].Length);

Это скопирует первую строку в один массив измерений.

я сделал метод расширения. я не знаю о производительности.

public static class ExtensionMethods 
{
     public static string[] get1Dim(this string[,] RectArr, int _1DimIndex , int _2DimIndex   )
     {
        string[] temp = new string[RectArr.GetLength(1)];

        if (_2DimIndex == -1)
        {
          for (int i = 0; i < RectArr.GetLength(1); i++)
          {   temp[i] = RectArr[_1DimIndex, i];    }
        }
        else
        {
          for (int i = 0; i < RectArr.GetLength(0); i++)
          {   temp[i] = RectArr[  i , _2DimIndex];    }
        }

         return temp;
      }
}

использование

// we now have this funtionaliy RectArray[1, * ]  
//                                       -1 means ALL    
string[] _1stRow = RectArray.get1Dim( 0, -1) ;    
string[] _2ndRow = RectArray.get1Dim( 1, -1) ; 

string[] _1stCol = RectArray.get1Dim( -1, 0) ;    
string[] _2ndCol = RectArray.get1Dim( -1, 1) ; 
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top