C #: изменение значений для каждого элемента в массиве - PullRequest
15 голосов
/ 06 октября 2010

Мне интересно, есть ли встроенная функциональность .NET для изменения каждого значения в массиве на основе результата предоставленного делегата. Например, если бы у меня был массив {1,2,3} и делегат, который возвращает квадрат каждого значения, я хотел бы иметь возможность запустить метод, который принимает массив и делегат и возвращает {1,4,9}. Что-нибудь подобное уже существует?

Ответы [ 6 ]

30 голосов
/ 06 октября 2010

LINQ обеспечивает поддержку проекций с использованием метода расширения Select :

var numbers = new[] {1, 2, 3};
var squares = numbers.Select(i => i*i).ToArray();

Вы также можете использовать чуть менее плавный Array.ConvertAll метод:

var squares = Array.ConvertAll(numbers, i => i*i);

Зубчатые массивы можно обработать, вложив проекции:

var numbers = new[] {new[] {1, 2}, new[] {3, 4}};
var squares = numbers.Select(i => i.Select(j => j*j).ToArray()).ToArray();

Многомерные массивы немного сложнее.Я написал следующий метод расширения, который проецирует каждый элемент в многомерном массиве независимо от его ранга.

static Array ConvertAll<TSource, TResult>(this Array source,
                                          Converter<TSource, TResult> projection)
{
    if (!typeof (TSource).IsAssignableFrom(source.GetType().GetElementType()))
    {
        throw new ArgumentException();
    }
    var dims = Enumerable.Range(0, source.Rank)
        .Select(dim => new {lower = source.GetLowerBound(dim),
                            upper = source.GetUpperBound(dim)});
    var result = Array.CreateInstance(typeof (TResult),
        dims.Select(dim => 1 + dim.upper - dim.lower).ToArray(),
        dims.Select(dim => dim.lower).ToArray());
    var indices = dims
        .Select(dim => Enumerable.Range(dim.lower, 1 + dim.upper - dim.lower))
        .Aggregate(
            (IEnumerable<IEnumerable<int>>) null,
            (total, current) => total != null
                ? total.SelectMany(
                    item => current,
                    (existing, item) => existing.Concat(new[] {item}))
                : current.Select(item => (IEnumerable<int>) new[] {item}))
        .Select(index => index.ToArray());
    foreach (var index in indices)
    {
        var value = (TSource) source.GetValue(index);
        result.SetValue(projection(value), index);
    }
    return result;
}

Вышеупомянутый метод может быть протестирован с массивом ранга 3 следующим образом:

var source = new int[2,3,4];

for (var i = source.GetLowerBound(0); i <= source.GetUpperBound(0); i++)
    for (var j = source.GetLowerBound(1); j <= source.GetUpperBound(1); j++)
        for (var k = source.GetLowerBound(2); k <= source.GetUpperBound(2); k++)
            source[i, j, k] = i*100 + j*10 + k;

var result = (int[,,]) source.ConvertAll<int, int>(i => i*i);

for (var i = source.GetLowerBound(0); i <= source.GetUpperBound(0); i++)
    for (var j = source.GetLowerBound(1); j <= source.GetUpperBound(1); j++)
        for (var k = source.GetLowerBound(2); k <= source.GetUpperBound(2); k++)
        {
            var value = source[i, j, k];
            Debug.Assert(result[i, j, k] == value*value);
        }
20 голосов
/ 06 октября 2010

Не то, чтобы я знал (заменяя каждый элемент вместо преобразования в новый массив или последовательность), но это невероятно легко написать:

public static void ConvertInPlace<T>(this IList<T> source, Func<T, T> projection)
{
    for (int i = 0; i < source.Count; i++)
    {
        source[i] = projection(source[i]);
    }
}

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

int[] values = { 1, 2, 3 };
values.ConvertInPlace(x => x * x);

Конечно, если на самом деле не нужно для изменения существующего массива, другие ответы, опубликованные с использованием Select, будут более функциональными. Или существующий метод ConvertAll из .NET 2:

int[] values = { 1, 2, 3 };
values = Array.ConvertAll(values, x => x * x);

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

5 голосов
/ 06 октября 2010

Используя System.Linq, вы можете сделать что-то вроде:

var newArray = arr.Select(x => myMethod(x)).ToArray();
2 голосов
/ 06 октября 2010

LINQ-запросы могут легко решить эту проблему для вас - убедитесь, что вы ссылаетесь на System.Core.dll и имеете оператор

using System.Linq;

.Например, если ваш массив находится в переменной с именем numberArray, следующий код даст вам именно то, что вы ищете:

 var squares = numberArray.Select(n => n * n).ToArray();

Последний вызов ToArray необходим только в том случае, если вам действительно нужномассив, а не IEnumerable .

1 голос
/ 03 августа 2016

Вот еще одно решение для массивов M x N, где M и N не известны во время компиляции.

    // credit: https://blogs.msdn.microsoft.com/ericlippert/2010/06/28/computing-a-cartesian-product-with-linq/
    public static IEnumerable<IEnumerable<T>> CartesianProduct<T>(IEnumerable<IEnumerable<T>> sequences)
    {
        IEnumerable<IEnumerable<T>> result = new[] { Enumerable.Empty<T>() };
        foreach (var sequence in sequences)
        {
            // got a warning about different compiler behavior
            // accessing sequence in a closure
            var s = sequence;
            result = result.SelectMany(seq => s, (seq, item) => seq.Concat<T>(new[] { item }));
        }
        return result;
    }


    public static void ConvertInPlace(this Array array, Func<object, object> projection)
    {
        if (array == null)
        {
            return;
        }

        // build up the range for each dimension
        var dimensions = Enumerable.Range(0, array.Rank).Select(r => Enumerable.Range(0, array.GetLength(r)));

        // build up a list of all possible indices
        var indexes = EnumerableHelper.CartesianProduct(dimensions).ToArray();

        foreach (var index in indexes)
        {
            var currentIndex = index.ToArray();
            array.SetValue(projection(array.GetValue(currentIndex)), currentIndex);
        }
    }
1 голос
/ 06 октября 2010

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

int[] x =  {1,2,3};
x = x.Select(( Y ) => { return Y * Y; }).ToArray();
...