¿Cómo iteras una matriz multidimensional sin saber el número de dimensiones y elementos de la matriz que se te pasa?

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

Pregunta

Un SDK me devuelve una matriz con varias dimensiones como:

   int [,,] theArray = new int [2,8,12];

Necesito visitar cada elemento de la matriz y devolver el valor y la posición del valor. Necesito hacer esto sin saber la cantidad de dimensiones y elementos de la matriz que se pasa.

¿Fue útil?

Solución

¿Te gustaría algo como esto? Recurre los rangos para que pueda usar un foreach () y obtener una matriz que contenga los índices del elemento actual.

class Program
{
    static void Main(string[] args)
    {
        int[, ,] theArray = new int[2, 8, 12];
        theArray[0, 0, 1] = 99;
        theArray[0, 1, 0] = 199;
        theArray[1, 0, 0] = 299;

        Walker w = new Walker(theArray);

        foreach (int i in w)
        {
            Console.WriteLine("Item[{0},{1},{2}] = {3}", w.Pos[0], w.Pos[1], w.Pos[2], i);
        }

        Console.ReadKey();
    }

    public class Walker : IEnumerable<int>
    {
        public Array Data { get; private set; }
        public int[] Pos { get; private set; }

        public Walker(Array array)
        {
            this.Data = array;
            this.Pos = new int[array.Rank];
        }

        public IEnumerator<int> GetEnumerator()
        {
            return this.RecurseRank(0);
        }

        private IEnumerator<int> RecurseRank(int rank)
        {
            for (int i = this.Data.GetLowerBound(rank); i <= this.Data.GetUpperBound(rank); ++i)
            {
                this.Pos.SetValue(i, rank);

                if (rank < this.Pos.Length - 1)
                {
                    IEnumerator<int> e = this.RecurseRank(rank + 1);
                    while (e.MoveNext())
                    {
                        yield return e.Current;
                    }
                }
                else
                {
                    yield return (int)this.Data.GetValue(this.Pos);
                }
            }
        }

        System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
        {
            return this.RecurseRank(0);
        }
    }
}

Otros consejos

Uso para bucles:

for (int i=theArray.GetLowerBound(0);i<=theArray.GetUpperBound(0);++i)
{
    for (int j=theArray.GetLowerBound(1);j<=theArray.GetUpperBound(1);++j)
    {
        for (int k=theArray.GetLowerBound(2);k<=theArray.GetUpperBound(2);++k)
        {
           // do work, using index theArray[i,j,k]
        }
    }
}

Si no conoce el número de dimensiones de antemano, puede usar Array.Rank para determinar eso.

No estoy seguro de entender su pregunta sobre " devolver la posición [n, n, n] " , pero si está intentando devolver más de un valor de un método , hay un par de formas de hacerlo.

• Utilice out o parámetros de referencia (por ejemplo, Int ) que se configuran con los valores devueltos antes de regresar del método.

• Pase una matriz, por ejemplo, una matriz de tres entradas, cuyos elementos se establecen mediante el método antes de que regrese.

• Devuelve una matriz de valores, por ejemplo, una matriz de tres ints.

Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top