Как получить полную строку или столбец из 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