我为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上进行分组吗?
答案 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()
};
这是输出:
更新:正如所承诺的,这是一个完整的工作示例:
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();
}