1. 程式人生 > >LINQ分組查詢—GroupBy()

LINQ分組查詢—GroupBy()

今天寫專案時遇到一個分組查詢的需求:將訂單列表中商品明細按商品編號彙總,我這裡簡單的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。