Вопрос

Мне нравится понимание списков в Python, потому что они кратко представляют преобразование списка.

Однако на других языках я часто ловлю себя на том, что пишу что-то вроде:

foreach (int x in intArray)
  if (x > 3) //generic condition on x
    x++ 
    //do other processing

Этот пример приведен на C #, где у меня сложилось впечатление, что LINQ может помочь с этим, но есть ли какая-нибудь распространенная программная конструкция, которая может заменить это несколько менее элегантное решение?Возможно, структура данных, которую я не рассматриваю?

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

Решение

Приращение в оригинале foreach цикл не повлияет на содержимое массива, единственным способом сделать это остается for петля:

for(int i = 0; i < intArray.Length; ++i)
{
    if(intArray[i] > 3) ++intArray[i];
}

Linq не предназначен для изменения существующих коллекций или последовательностей.Он создает новые последовательности на основе существующих.Приведенный выше код можно реализовать с помощью Linq, хотя это немного противоречит его целям:

var newArray1 = from i in intArray select ((i > 3) ? (i + 1) : (i));
var newArray2 = intArray.Select(i => (i > 3) ? (i + 1) : (i));

Используя where (или эквивалентный), как показано в некоторых других ответах, исключит любые значения, меньшие или равные 3, из результирующей последовательности.

var intArray = new int[] { 10, 1, 20, 2 };
var newArray = from i in intArray where i > 3 select i + 1;
// newArray == { 11, 21 }

Существует ForEach метод для массивов, который позволит вам использовать лямбда-функцию вместо foreach блокировать, хотя для чего-либо большего, чем вызов метода, я бы придерживался foreach.

intArray.ForEach(i => DoSomething(i));

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

В C # вы можете применить выборочную обработку ко всему, что находится внутри IEnumerable, например:

intArray.Where(i => i > 3).ConvertAll();
DoStuff(intArray.Where(i => i 3));

И т.д..

В Python у вас есть фильтр и сопоставление, который может сделать так, как вы хотите:

map(lambda x: foo(x + 1) filter(lambda x: x > 3, intArray))

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

[f(x + 1) for x in intArray if x > 3]

в Ruby:

intArray.select { |x| x > 3 }.each do |x|
  # do other processing
end

или, если "другая обработка" - это короткая однострочная:

intArray.select { |x| x > 3 }.each { |x| something_that_uses x }

наконец, если вы хотите вернуть новый массив, содержащий результаты обработки тех элементов, размер которых превышает 3:

intArray.select { |x| x > 3 }.map { |x| do_something_to x }
map(lambda x: test(x + 1) filter(lambda x: x > 3, arr))

В зависимости от языка и того, что вам нужно сделать, "карта", как она называется на многих языках, может быть тем, что вы ищете.Я не знаю C #, но согласно это страница .NET 2.0 вызывает map "ConvertAll".

Значение "map" довольно простое - возьмите список и примените функцию к каждому его элементу, возвращая новый список.Возможно, вы также ищете "фильтр", который выдаст вам список элементов, удовлетворяющих предикату в другом списке.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top