Как получить полную строку или столбец из 2D-массива в С#

Я не хочу использовать зубчатый массив, у меня есть двумерный массив, и я хочу получить полный столбец или строку, не проходя через них. У кого-нибудь есть идеи, как это можно сделать.

double [,]  array = new double [3,3] ;

1   2   3 
4   5   6

Out: 1   2   3  or 2   5 

Ответ 1

Вы можете оптимизировать его для получения строк, используя Buffer.BlockCopy(), но чтобы получить столбец, вам придется выполнить цикл. Buffer.BlockCopy() в конечном итоге использует инструкцию процессора для копирования блока памяти, поэтому он довольно быстрый.

Удобно поместить код в метод расширения, чтобы его было проще вызывать. Обратите внимание, что Buffer.BlockCopy() может использоваться только для массивов примитивных типов, то есть int, double, char и т.д. Это НЕ включает string.

Вот скомпилированный пример:

using System;
using System.Linq;
using System.Runtime.InteropServices;

namespace ConsoleApplication4
{
    public static class Program
    {
        private static void Main()
        {
            var array = new [,]
            {
                {0.1, 0.2, 0.3, 0.4, 0.5},
                {1.1, 1.2, 1.3, 1.4, 1.5},
                {2.1, 2.2, 2.3, 2.4, 2.5},
                {3.1, 3.2, 3.3, 3.4, 3.5},
            };

            var row = array.GetRow(2);

            // This prints 2.1, 2.2, 2.3, 2.4, 2.5

            Console.WriteLine(string.Join(", ", row.Select(element => element.ToString())));
        }
    }

    public static class ArrayExt
    {
        public static T[] GetRow<T>(this T[,] array, int row)
        {
            if (!typeof(T).IsPrimitive)
                throw new InvalidOperationException("Not supported for managed types.");

            if (array == null)
                throw new ArgumentNullException("array");

            int cols = array.GetUpperBound(1) + 1;
            T[] result = new T[cols];

            int size;

            if (typeof(T) == typeof(bool))
                size = 1;
            else if (typeof(T) == typeof(char))
                size = 2;
            else
                size = Marshal.SizeOf<T>();

            Buffer.BlockCopy(array, row*cols*size, result, 0, cols*size);

            return result;
        }
   }
}

Ответ 2

Чтобы получить определенную строку или столбец из многомерного массива, вы можете использовать LINQ:

public class CustomArray<T>
{
    public T[] GetColumn(T[,] matrix, int columnNumber)
    {
        return Enumerable.Range(0, matrix.GetLength(0))
                .Select(x => matrix[x, columnNumber])
                .ToArray();
    }

    public T[] GetRow(T[,] matrix, int rowNumber)
    {
        return Enumerable.Range(0, matrix.GetLength(1))
                .Select(x => matrix[rowNumber, x])
                .ToArray();
    }
}

Ответ 3

Я не думаю, что есть способ без цикла. Это может быть сделано и с LINQ, но это заставляет вас за него

Ответ 4

В компьютерах нет чудес.
Это то, что вы хотите?

class ColumnAccessor<T> where T : class
{
    int columnIndex;
    T[,] arr;
    public ColumnAccessor(T[,] arr, int columnIndex)
    {
        this.columnIndex = columnIndex;
        this.arr = arr;
    }
    public T Get(int rowIndex)
    {
        return this.arr[columnIndex, rowIndex];
    }
}

Ответ 5

Вот как я это сделал, вы можете использовать

GetLength(0)

чтобы получить столбцы и использовать

GetLength(1)

чтобы получить строки 2-мерного массива, и вы перебираете его через цикл for, если кому-то еще это нужно.

string text = "";
for (int i = 0; i < array.GetLength(0); i++)
   {
      text += Convert.ToString(array[i, 2]) + "\n";
   }

Ответ 6

альтернативный способ, которым вы можете это сделать, это использовать Список вместо массива.

В частности, в вашем случае вы бы сделали что-то подобное:

  • Сначала создайте внутренний класс, представляющий кортеж массива
  • Создать список внутреннего класса
  • Заполните внутренний класс
  • Получить строку, содержащую что-то конкретное
  • Получить столбец, содержащий что-то конкретное
public static void Main(string[] args)
{
    // #2 -- Instantiate List of myClass
    List<myClass> myList = new List<myClass>();
    //
    // #3 -- Populate the list
    myList.Add(new myClass(1,2,3));           
    myList.Add(new myClass(3,4,5));
    myList.Add(new myClass(5,6,6));
    //
    // #4 -- Get the line where a == 1
    myList.Find(x=>x.a == 1);
    //
    // #5 -- Get column b
    myList.Select(x=>x.b);
}
// #1 -- Create the inner class
public class myClass
{
    public int a;
    public int b;
    public int c;
    public myClass(int a, int b, int c)
    {
        this.a =a;
        this.b =b;
        this.c =c;
    }
}

Ответ 7

что нужно, так это массив Jugged (не многомерный массив)

https://msdn.microsoft.com/en-us/library/2s05feca.aspx

int[][] jaggedArray = new int[3][];
jaggedArray[0] = new int[5];
jaggedArray[1] = new int[] { 0, 2, 4, 6 };
jaggedArray[2] = new int[] { 11, 22 };

полный пример со столбцами:

using System;
using System.Collections.Generic;

namespace Rextester
{
    public class Program
    {

        public static T[] column<T>(T[][] jaggedArray,int wanted_column)
        {
            T[] columnArray = new T[jaggedArray.Length];
            T[] rowArray;
            for(int i=0;i<jaggedArray.Length;i++)
            {
                rowArray=jaggedArray[i];
                if(wanted_column<rowArray.Length)
                    columnArray[i]=rowArray[wanted_column];
            }
            return columnArray;
        }

        public static void Main(string[] args)
        {
            //Your code goes here
                int[][] jaggedArray = new int[3][];
                jaggedArray[0] = new int[5];
                jaggedArray[1] = new int[] { 0, 2, 4, 6 };
                jaggedArray[2] = new int[] { 11, 22 };

                Console.WriteLine("Hello, world!");
                Console.WriteLine(string.Join(" ",jaggedArray[1]));
                Console.WriteLine(string.Join(" ",column(jaggedArray,1)));
        }
    }
}

другая та же идея, используя расширения

using System;
using System.Collections.Generic;

namespace Rextester
{
    public static class MyExtensions
    {
        public static string Extend(this Array array)
        {
            return "Yes, you can extend an array";
        }

        public static T[] column<T>(this T[,] multidimArray,int wanted_column)
        {
            int l=multidimArray.GetLength(0);
            T[] columnArray = new T[l];
            for(int i=0;i<l;i++)
            {
              columnArray[i]=multidimArray[i,wanted_column];
            }
            return columnArray;
        }

        public static T[] row<T>(this T[,] multidimArray,int wanted_row)
        {
            int l=multidimArray.GetLength(1);
            T[] rowArray = new T[l];
            for(int i=0;i<l;i++)
            {
              rowArray[i]=multidimArray[wanted_row,i];
            }
            return rowArray;
        }


    } 

    public class Program
    {


        public static void Main(string[] args)
        {
                Console.WriteLine("Hello, world!");

                int [,] multidimArray = new int[,] { { 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 } };
                Console.WriteLine(string.Join(" ",multidimArray.column(0)));
                Console.WriteLine(string.Join(" ",multidimArray.row(0)));

        }
    }
}

Ответ 8

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

double[,] array = new double[3,3] {{1,2,3}, {4,5,6}, {7,8,9}}; 

int firstNum = array[0,1];
int secondNum = array[1,1];

получится 2, 5