GroupBy与Linq to Objects

时间:2011-08-09 06:05:09

标签: group-by linq-to-objects

我为Linq打了一个GroupBy。我想我不能在一个表达式中做到这一点,但不知道如何解决问题。

我有以下两个班级:

public class Shipment {
  public string PortOfOrigin{get;set;}
  public string PortOfDestination{get;set;}
  public ICollection<Invoice> Invoices{get;set;}
}

public class Invoice{
  public string InvoicePeriod {get;set;}
  public decimal Amount {get;set;}
}

我有一系列发货,都有一个发票集合。这是一个例子。

List<Shipment> shipments = new List<Shipment>
{
    new Shipment { PortOfOrigin = "USLOS", PortOfDestionation = "UKLON", 
        Invoices = new List<Invoice>
        {
            new Invoice{InvoicePeriod = "201106", Amount = 1000},
            new Invoice{InvoicePeriod = "201106", Amount = 2000},
            new Invoice{InvoicePeriod = "201107", Amount = 1000}
        }
    },
    new Shipment { PortOfOrigin = "USLOS", PortOfDestionation = "UKLON", 
        Invoices = new List<Invoice>
        {
            new Invoice{InvoicePeriod = "201106", Amount = 3000},
            new Invoice{InvoicePeriod = "201107", Amount = 2000}
        }
    },
    new Shipment { PortOfOrigin = "USDAL", PortOfDestionation = "UKLON", 
        Invoices = new List<Invoice>
        {
            new Invoice{InvoicePeriod = "201106", Amount = 3000}
        }
    }
};

现在我想通过以下方式进行分组:

Shipment.PortOfOrigin,Shipment.PortOfDestionation,Shipment.Invoices.InvoicePeriod。

所以我想要这样的结果

PortOfOrigin    PortOfDestination   InvoicePeriod   Amount
------------------------------------------------------------------------
USLOS           UKLON               201106          6000
USLOS           UKLON               201107          3000
USDAL           UKLON               201106          3000

这甚至可以做一个这样的组,我想在Invoices.InvoicePeriod上进行分组吗?

1 个答案:

答案 0 :(得分:1)

更新:我已更新我的回答以包含发货数量。这要求货件有一个ID字段。


以下将解决这个问题:

//first flatten the data so it's easier to manipulate
var query = from s in shipments
            from i in s.Invoices
            select new
            {
                s.ShipmentUniqueIdentifier,
                s.PortOfOrigin,
                s.PortOfDestination,
                i.InvoicePeriod,
                i.Amount
            };

//group the data as desired
var grouping = query.GroupBy(q => new { q.PortOfOrigin, q.PortOfDestination, q.InvoicePeriod });

//finally sum the amounts
var results = from g in grouping
              select new
              {
                  g.Key.PortOfOrigin,
                  g.Key.PortOfDestination,
                  g.Key.InvoicePeriod,
                  Amount = g.Select(s => s.Amount).Sum(),
                  NumberOfShipments = g.Select(s => s.ShipmentUniqueIdentifier).Distinct().Count()
              };

这是输出:

enter image description here

更新:正如所承诺的,这是一个完整的工作示例:

public class Shipment {
  public Guid ShipmentUniqueIdentifier{get;set;}
  public string PortOfOrigin{get;set;}
  public string PortOfDestination{get;set;}
  public ICollection<Invoice> Invoices{get;set;}
}

public class Invoice {
  public string InvoicePeriod {get;set;}
  public decimal Amount {get;set;}
}

List<Shipment> shipments = new List<Shipment>
{
    new Shipment { PortOfOrigin = "USLOS", PortOfDestination = "UKLON", ShipmentUniqueIdentifier = Guid.NewGuid(),
        Invoices = new List<Invoice>
        {
            new Invoice{InvoicePeriod = "201106", Amount = 1000},
            new Invoice{InvoicePeriod = "201106", Amount = 2000},
            new Invoice{InvoicePeriod = "201107", Amount = 1000}
        }
    },
    new Shipment { PortOfOrigin = "USLOS", PortOfDestination = "UKLON", ShipmentUniqueIdentifier = Guid.NewGuid(),
        Invoices = new List<Invoice>
        {
            new Invoice{InvoicePeriod = "201106", Amount = 3000},
            new Invoice{InvoicePeriod = "201107", Amount = 2000}
        }
    },
    new Shipment { PortOfOrigin = "USDAL", PortOfDestination = "UKLON", ShipmentUniqueIdentifier = Guid.NewGuid(), 
        Invoices = new List<Invoice>
        {
            new Invoice{InvoicePeriod = "201106", Amount = 3000}
        }
    }
};

void Main()
{
    //first flatten the data so it's easier to manipulate
    var query = from s in shipments
                from i in s.Invoices
                select new
                {
                    s.ShipmentUniqueIdentifier,
                    s.PortOfOrigin,
                    s.PortOfDestination,
                    i.InvoicePeriod,
                    i.Amount
                };

    //group the data as desired
    var grouping = query.GroupBy(q => new { q.PortOfOrigin, q.PortOfDestination, q.InvoicePeriod });

    //finally sum the amounts
    var results = from g in grouping
                  select new
                  {
                      g.Key.PortOfOrigin,
                      g.Key.PortOfDestination,
                      g.Key.InvoicePeriod,
                      Amount = g.Select(s => s.Amount).Sum(),
                      NumberOfShipments = g.Select(s => s.ShipmentUniqueIdentifier).Distinct().Count()
                  };

    //output the results
    results.Dump();
}