Операторы c# Linq или foreach() для суммирования подмножеств?

какое из этих решений предпочтительнее?

список:

List<ExampleInfo> exampleList = new List<ExampleInfo>();

public class ExampleInfo
{
    internal ExampleInfo()
    { }
    /* Business Properties */
    public int Id { get; set; }
    public string Type { get; set; }
    public decimal Total { get; set; }
}

Я хочу получить промежуточные итоги на основе "общего" значения.

Вариант 1:

var subtotal1 = exampleList.Where(x => x.Type == "Subtype1").Sum(x => x.Total);
var subtotal2 = exampleList.Where(x => x.Type == "Subtype2").Sum(x => x.Total);

Вариант 2:

decimal subtotal1 = 0m;
decimal subtotal2 = 0m;
foreach (ExampleInfo example in exampleList)
{
    switch (example.Type)
    {
        case "Subtype1":
            subtotal1 += example.Total;
             break;
        case "Subtype2":
             subtotal2 += example.Total;
             break;
        default:
             break;

    }
}

список будет

Edit: Крис поднял очень хороший вопрос, о котором я не упомянул. Программа уже использует .NET Framework 3.5 SP1, поэтому совместимость не является важным фактором здесь.

5 ответов


оба этих примера имеют дублированный код, и оба не готовы к изменению на Type - что, если бы у него было три значения? А если бы их было 30?
Вы можете использовать linq для группировки по нему и получить общее:

var totals = from p in exampleList
             group p by p.Type into g
             select new { Type = g.Key, Total = g.Sum(p => p.Total ) };

так totals представляет собой набор объектов со свойствами Type и Total


независимо от размера списка, если вы нацелены на .NET 3.5, я бы пошел с LINQ, хотя бы для удобства чтения.

Я большой поклонник написания того, что вы имеете в виду, а не как это делается, и LINQ делает это очень легко в таких случаях.

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

var subtotals = from x in exampleList
                group x by x.Type into g
                select new { Type = x.Key, SubTotal = g.Sum(x => x.Total) };

(Не совсем уверен, работает ли код, как это, это просто быстрая адаптация от одного из 101 LINQ Samples. Синтаксис должен быть в порядке.)


3

var groupings = exampleList
    .GroupBy(x => x.Type, x => x.Total)
    .Select(x => new { Type = x.Key, SubTotal = x.Sum() } );

у вас будет список таких классов:

class <Anonymous>
{
    public string Type { get; }    
    public decimal SubTotal { get; }  
}

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


Я не думаю, что для таких маленьких списков будет большая разница в производительности.

опция 1 повторит список дважды, в то время как опция 2 повторяет список только один раз. Это может быть более важно отметить для больших списков, чем для небольших.

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

очевидным преимуществом варианта 2 является то, что код работает в .NET Framework версии 2.0. Использование LINQ означает, что вашему приложению требуется .NET Framework 3.5.


для option1 внутренний цикл foreach будет выполняться дважды env времени выполнения C#. Следовательно, тип обработки будет больше. Но для