今天写项目时遇到一个分组查询的需求:将订单列表中商品明细按商品编号汇总,我这里简单的mark一下。
之所以要记录,是因为之前很少用linq去写分组查询,其次是在此过程中遇到了一个小问题。
我们都知道linq语句最后也是要转化成sql语句的,很多人忽略开发效率一味地认为linq的执行效率没有sql的高,或者因为习惯问题不愿意使用linq。(今天在此就不探讨这个问题了,哲学思想告诉我“存在即合理”!)下面可以根据代码直观地感受一下两者的区别。
/// <summary>
/// 读取对账单的订单明细的汇总
/// </summary>
/// <param name="statement"></param>
public IList<OrderDetailCollect> LoadOrderCollect(OrderStatement statement)
{
BaseRepo<OrderDetailCollect> br = new BaseRepo<OrderDetailCollect>();
string sql = string.Format(@" select vod.GoodsId,vod.DisplayName, vod.Unit,Sum(vod.Num) as Num,vod.Price,Sum(vod.Amount) as Amount
,Sum(vod.DisplayNum) as DisplayNum,vod.DisplayUnit, vod.DisplayPrice
,ISNULL((select SUM(SumMoney) from Orders where Id In (select OrderId from OrderStatementDetail where OrderStatementId = {0} )),0) as TotalAmount
from View_OrderDetail vod inner join OrderStatementDetail osd on vod.OrderId = osd.OrderId
where osd.OrderStatementId = {0} group by vod.GoodsId, vod.DisplayName, vod.DisplayUnit, vod.DisplayPrice, vod.Unit, vod.Price", statement.Id);
var res = br.SqlQuery<OrderDetailCollect>(sql).Items;
return res;
}
当然,这段代码本身没有什么大问题,但是你要后面加入进来的开发者 来维护一段sql,可能就要发不少时间整理了。再加上个人有强迫症,页面上出现大段大段的sql语句我会很不舒服。
//待投影的model
public class GoodsDetailCollect
{
public int GoodsId { get; set; }
public string DisplayName { get; set; }
public decimal Amount { get; set; }
public int DisplayNum { get; set; }
public decimal DisplayPrice { get; set; }
public string DisplayUnit { get; set; }
}
//扩展方法GroupBy()
var goodsCollect = details.GroupBy(d => d.GoodsId).Select(g => (new GoodsDetailCollect()//这里的details是未分组前的一个list
{
GoodsId = g.Key,
DisplayNum = g.Sum(n => n.Num),
DisplayName = g.Select(n => n.DisplayName).FirstOrDefault() ?? "",
DisplayUnit = g.Select(n => n.Unit).FirstOrDefault() ?? "",
DisplayPrice = g.Select(n => n.Price).FirstOrDefault(),
Amount = g.Sum(a => a.Amount)
})).ToList();
//var goodsCollect2 = (from detail in details
// group detail by detail.GoodsId into d
// select new GoodsDetailCollect()
// {
// GoodsId = d.Key,
// DisplayNum = d.Sum(n => n.Num),
// DisplayName = d.Select(n => n.DisplayName).FirstOrDefault() ?? "",
// DisplayUnit = d.Select(n => n.Unit).FirstOrDefault() ?? "",
// DisplayPrice = d.Select(n => n.Price).FirstOrDefault(),
// Amount = d.Sum(a => a.Amount)
// }).ToList();
遇到的问题:
分组之后发现,要得到这个键或者对序列中的一些字段进行统计或者计算(比如:最大值/最小值/平均值/求和等等)很简单,但是要投影model时,怎么拿这个序列里面具体的某个值呢?
看一下“g”的类型===》,然而对此并不是太了解。
于是查看微软文档:戳这里移步微软文档,原来根据GoodsId属性值分组后,details对象中GoodsId的每个唯一值都成为新IGrouping <int,View_PurchaseStatementGoodsDetail>对象的键,并且具有该键的details对象形成IGrouping <TKey,View_PurchaseStatementGoodsDetail>对象的值序列。而这里的IGrouping <TKey,View_PurchaseStatementGoodsDetail>对象就是“g”。因此,要访问值序列,只需使用“g”变量本身即可。
最后,人生苦短,我用LINQ。