欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C# GroupBy的基本使用教程

 更新時(shí)間:2019年02月01日 09:51:58   作者:weilence  
這篇文章主要給大家介紹了關(guān)于C# GroupBy的基本使用教程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

起因

今天在公司做一個(gè)需求的時(shí)候,寫的是面條代碼,一個(gè)方法直接從頭寫到尾,其中用到了GroupBy,且GroupBy的KeySelector是多個(gè)屬性而不是單個(gè)屬性。

但是公司最近推行Clean Code,要讓代碼有可讀性。且作為一個(gè)有追求的程序員,肯定是不能寫面條代碼的,要對(duì)代碼進(jìn)行拆分。

重構(gòu)前GroupBy大概是這樣子的:

var groups = data.GroupBy(m => new { m.PropertyA, m.PropertyB})

個(gè)人對(duì)于短的Linq比較習(xí)慣于用方法而不是用關(guān)鍵字的那種寫法。

一開始這樣寫是沒問題的,但是重構(gòu)的時(shí)候問題就來了:這個(gè)groups是什么類型?

重構(gòu)以后這個(gè)groups是要作為參數(shù)進(jìn)入到別的方法中的,方法簽名顯然是不能用var做類型推導(dǎo),必須指定確定的類型。

我們知道GroupBy出來的東西是個(gè)泛型的東西,簽名是IEnumerable<IGrouping<TKey, TSource>>,這個(gè)TSource類型是沒問題,我沒有對(duì)Source做修改,就是data本身的類型。

但是這個(gè)Key就有問題了。

我沒有指定Key的類型,這里應(yīng)該是匿名類型,于是定義了一個(gè)類型承接Key,代碼變成了:

class EntityKey
{
 public int PropertyA { get set; }
 public string PropertyB { get set; }
}

......

var groups = data.GroupBy(m => new EntityKey { PropertyA = m.PropertyA, PropertyB = m.PropertyB});

但是后來我發(fā)現(xiàn)這樣有問題,GroupBy指定的Key失效了。也就是說,groups的分組數(shù)量與data的長(zhǎng)度一致,每一個(gè)group里面只有一個(gè)對(duì)象。

分析

發(fā)現(xiàn)這個(gè)問題后,我仔細(xì)思考了一下,大致猜到了問題出在哪里。

GroupBy這種東西,判斷兩個(gè)對(duì)象是不是一個(gè)分組,必然用到了相等判斷。

雖然我沒有看匿名類型反編譯生成后的IL代碼,不知道之前用的是怎么做的Key相等判斷,但是引用類型的肯定是直接用對(duì)象的HashCode做判斷。

這樣子肯定是不行的,要解決引用類型的相等判斷問題。

重現(xiàn)

根據(jù)猜測(cè),我寫了一個(gè)Sample程序最小化的重現(xiàn)了這個(gè)問題:

class Program
{
 static void Main(string[] args)
 {
  var list = new List<Student>();
  list.Add(new Student(1, "Cat", 10, "University1"));
  list.Add(new Student(2, "Dog", 10, "University1"));
  list.Add(new Student(3, "Pig", 10, "University2"));
  list.Add(new Student(4, "Fish", 12, "University1"));

  var groups = list.GroupBy(m => new {m.Age, m.Class});
  
  foreach (var group in groups)
  {
   Console.WriteLine("Age:{0},Class:{1}", group.Key.Age, group.Key.Class);
   foreach (var student in group)
   {
    Console.WriteLine(student);
   }
  }
 }

 class Student
 {
  public int Id { get; set; }
  public string Name { get; set; }
  public int Age { get; set; }
  public string Class { get; set; }

  public Student(int id, string name, int age, string @class)
  {
   Id = id;
   Name = name;
   Age = age;
   Class = @class;
  }

  public override string ToString()
  {
   return $"Id={Id},Name={Name},Age={Age},Class={Class}";
  }
 }

 class StudentKey
 {
  public int Age { get; set; }
  public string Class { get; set; }
 }
}

這時(shí)候輸出結(jié)果是

Age:10,Class:University1
Id=1,Name=Cat,Age=10,Class=University1
Id=2,Name=Dog,Age=10,Class=University1
Age:10,Class:University2
Id=3,Name=Pig,Age=10,Class=University2
Age:12,Class:University1
Id=4,Name=Fish,Age=12,Class=University1

將new {m.Age, m.Class}替換為new StudentKey {Age = m.Age, Class = m.Class},結(jié)果卻變成了

Age:10,Class:University1
Id=1,Name=Cat,Age=10,Class=University1
Age:10,Class:University1
Id=2,Name=Dog,Age=10,Class=University1
Age:10,Class:University2
Id=3,Name=Pig,Age=10,Class=University2
Age:12,Class:University1
Id=4,Name=Fish,Age=12,Class=University1

Id=1和Id=2變成了兩組。

解決問題

解決問題方式有幾種。

第一種

最簡(jiǎn)單,就是直接將StudentKey從class變成struct。

但是這樣有個(gè)問題,class是堆內(nèi)存,struct是棧內(nèi)存。

雖然實(shí)際情況不一定會(huì)出現(xiàn)內(nèi)存異常什么的,但是總歸是改變了一些東西,存在隱患。

第二種

第一種方式被我自己否決后,于是打開了Google搜了一下,在StackOverflow和MSDN以及查看GroupBy源碼之后,得到了GroupBy的運(yùn)行原理。

GroupBy在沒有傳comparer的時(shí)候,會(huì)創(chuàng)建一個(gè)基于當(dāng)前TSource類型的默認(rèn)的comparer。

但不管是默認(rèn)的comparer還是我們自己傳的comparer,都會(huì)調(diào)用Equals和GetHashCode兩個(gè)方法,所以我們需要重載這兩個(gè)方法。

第二種方法就是我們?cè)陬愋蜕现剌dEquals和GetHashCode兩個(gè)方法。

可以實(shí)現(xiàn)IEquatable<TKey>使用下面的代碼,也可以不實(shí)現(xiàn)接口,使用重載的Equals方法。

但是不論如何,一定要重載GetHashCode。

修改后StudentKey如下

class StudentKey : IEquatable<StudentKey>
{
  public int Age { get; set; }
  public string Class { get; set; }

  public override int GetHashCode()
  {
    return Age.GetHashCode() ^ Class.GetHashCode();
  }
  
//      public override bool Equals(object obj)
//      {
//        var model = obj as StudentKey;
//        if (model == null)
//        {
//          return false;
//        }
//
//        return model.Age == Age && model.Class == Class;
//      }

  public bool Equals(StudentKey other)
  {
    return Age == other.Age && Class == other.Class;
  }
}

第三種

第三種就是傳一個(gè)comparer給GroupBy參數(shù),實(shí)現(xiàn)一個(gè)IEqualityComparer<TKey>。

代碼如下:

list.GroupBy(m => new StudentKey {Age = m.Age, Class = m.Class}, new StudentKeyComparer());

......

class StudentKeyComparer: IEqualityComparer<StudentKey>
{
  public bool Equals(StudentKey x, StudentKey y)
  {
    return x.Age == y.Age && x.Class == y.Class;
  }

  public int GetHashCode(StudentKey obj)
  {
    return obj.Age.GetHashCode() ^ obj.Age.GetHashCode();
  }
}

這種相對(duì)于第二種方式,最大的區(qū)別在于不用侵入實(shí)體類添加代碼,但是原理是類似的。

總結(jié)

本文是在c#開發(fā)過程中碰到的一個(gè)GroupBy的分組的Key失效的問題。

了解其分組原理后,通過實(shí)現(xiàn)Equals和GetHashCode或者傳入自定義的comparer,解決GroupBy的分組Key失效的問題。

好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • 基于C#技術(shù)實(shí)現(xiàn)身份證識(shí)別功能

    基于C#技術(shù)實(shí)現(xiàn)身份證識(shí)別功能

    這篇文章主要介紹了基于C#技術(shù)實(shí)現(xiàn)身份證識(shí)別功能的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2016-07-07
  • C#滑動(dòng)驗(yàn)證碼拼圖驗(yàn)證功能實(shí)現(xiàn)(SlideCaptcha)

    C#滑動(dòng)驗(yàn)證碼拼圖驗(yàn)證功能實(shí)現(xiàn)(SlideCaptcha)

    目前網(wǎng)站上的驗(yàn)證碼機(jī)制可謂是五花八門,有簡(jiǎn)單的數(shù)字驗(yàn)證,有摻雜了字母和文字的混淆驗(yàn)證,還有通過滑塊進(jìn)行的拼圖驗(yàn)證,下面這篇文章主要給大家介紹了關(guān)于C#滑動(dòng)驗(yàn)證碼拼圖驗(yàn)證功能的實(shí)現(xiàn)方法,需要的朋友可以參考下
    2022-04-04
  • C#使用Parallel類進(jìn)行多線程編程實(shí)例

    C#使用Parallel類進(jìn)行多線程編程實(shí)例

    這篇文章主要介紹了C#使用Parallel類進(jìn)行多線程編程的方法,實(shí)例分析了Parallel類的相關(guān)使用技巧,需要的朋友可以參考下
    2015-06-06
  • c#中Empty()和DefalutIfEmpty()用法分析

    c#中Empty()和DefalutIfEmpty()用法分析

    這篇文章主要介紹了c#中Empty()和DefalutIfEmpty()用法,以實(shí)例形式分析了針對(duì)不同情況下Empty()和DefalutIfEmpty()用法區(qū)別,需要的朋友可以參考下
    2014-11-11
  • 詳解CLR的內(nèi)存分配和回收機(jī)制

    詳解CLR的內(nèi)存分配和回收機(jī)制

    本文詳細(xì)講解了CLR的內(nèi)存分配和回收機(jī)制,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-03-03
  • c# 實(shí)現(xiàn)子窗口關(guān)閉父窗口也關(guān)閉的簡(jiǎn)單實(shí)例

    c# 實(shí)現(xiàn)子窗口關(guān)閉父窗口也關(guān)閉的簡(jiǎn)單實(shí)例

    下面小編就為大家?guī)硪黄猚# 實(shí)現(xiàn)子窗口關(guān)閉父窗口也關(guān)閉的簡(jiǎn)單實(shí)例。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-02-02
  • c# 進(jìn)程之間的線程同步

    c# 進(jìn)程之間的線程同步

    這篇文章主要介紹了c# 進(jìn)程之間的線程同步,幫助大家更好的理解和學(xué)習(xí)c#,感興趣的朋友可以了解下
    2020-10-10
  • C#開源的AOP框架--KingAOP基礎(chǔ)

    C#開源的AOP框架--KingAOP基礎(chǔ)

    這篇文章主要介紹了一款C#開源的AOP框架--KingAOP框架的基礎(chǔ)知識(shí),對(duì)于想學(xué)習(xí)AOP的小伙伴來說,非常不錯(cuò),希望大家能夠喜歡。
    2015-12-12
  • C#通過System.CommandLine快速生成支持命令行的應(yīng)用程序

    C#通過System.CommandLine快速生成支持命令行的應(yīng)用程序

    這篇文章介紹了C#通過System.CommandLine快速生成支持命令行應(yīng)用程序的方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • C#最小二乘法擬合曲線成直線的實(shí)例

    C#最小二乘法擬合曲線成直線的實(shí)例

    這篇文章主要介紹了C#最小二乘法擬合曲線成直線的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02

最新評(píng)論