当前位置: 移动技术网 > IT编程>开发语言>.net > C#在LINQ中使用GroupBy

C#在LINQ中使用GroupBy

2020年08月08日  | 移动技术网IT编程  | 我要评论
一、先准备要使用的类:1、person类:class person { public string name { set; get; } public int age { set; ge

一、先准备要使用的类:

1、person类:

class person
  {
    public string name { set; get; }
    public int age { set; get; }
    public string gender { set; get; }
    public override string tostring() => name;
  }

2、准备要使用的list,用于分组(groupby):

list<person> personlist = new list<person>
    {
      new person
      {
        name = "p1", age = 18, gender = "male"

      },
      new person
      {
        name = "p2", age = 19, gender = "male",
      },
      new person
      {
        name = "p2", age = 17,gender = "female",
      }
    };

二、第一种用法:

public static ienumerable<igrouping<tkey, tsource>> groupby<tsource, tkey>(this ienumerable<tsource> source, func<tsource, tkey> keyselector);

官方释义:根据指定的键选择器函数对序列中的元素进行分组。

我们要分组的集合为source,集合内每个元素的类型为tsource,这里第一个参数keyselector的类型为func<tsource, tkey>,用于将tsource元素按照由此委托返回的类型tkey进行分组,结果为一个已分好组的集合(集合中的集合)。

编写客户端试验代码如下:

var groups = personlist.groupby(p => p.gender);
    foreach (var group in groups)
    {
      console.writeline(group.key);
      foreach(var person in group)
      {
        console.writeline($"\t{person.name},{person.age}");
      }
    }

以上代码指定的keyselector是person类的gender属性,因此,以上会按照gender(性别)进行分组,我们使用两个嵌套的foreach循环将分组的内容打印到控制台。

因为groups返回的类型为ienumerable<igouping<tkey,tsource>>,因此以上返回的类型为ienumerable<igouping<string,person>>。

igouping<string,person>是已经分组后的集合,内部集合元素为person,且igouping有一个key属性,类型为string(指的是gender属性类型),用于分组的标识。

输出结果如下:

其等价的linq语句为:

var groups = from p in personlist
       group p by p.gender;

以上的意思可以这样理解:从personlist取出p,并对p进行分组,使用分组的依据(key)为p.gender,并将分组的结果存储到pgroup,并将分组的结果选择出来合并成一个集合。

三、第二种用法:

public static ienumerable<igrouping<tkey, tsource>> groupby<tsource, tkey>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, iequalitycomparer<tkey> comparer);

官方释义:根据指定的键选择器函数对序列中的元素进行分组,并使用指定的比较器对键进行比较。

这种比第一种方法多了一个参数,那就是一个相等比较器,目的是为了当tkey为自定义的类时,groupby能根据tkey指定的类根据相等比较器进行分组,

因此,自定义类如何进行分组,groupby是不知道的,需要自己定义自己的相等比较器。

首先,将personlist更改如下(下划线部分):

list<person> personlist = new list<person>
    {
      new person
      {
        name = "p1", age = 18, gender = "male"

      },
      new person
      {
        name = "p1", age = 19, gender = "male",
      },
      new person
      {
        name = "p3", age = 17,gender = "female",
      }
    };

其次,增加一个相等比较器类,用于对person进行分组:

  class personequalitycomparer : iequalitycomparer<person>
  {
    public bool equals(person x, person y) => x.name == y.name;
    public int gethashcode(person obj) => obj.name.gethashcode();
  }

其中定义了如何对一个person相等性定义,只要实现iequalitycomparer<person>即可,这里以name作为person类是否相同的依据。

最后,现在我们对person类进行分组,编写客户端实验代码如下:

var groups = personlist.groupby(p => p, new personequalitycomparer());
    foreach (var group in groups)
    {
      console.writeline(group.key.tostring());
      foreach(var person in group)
      {
        console.writeline($"\t{person.age},{person.gender}");
      }
    }

以上的分组依据是person类,并运用了自己定义的person类相同比较器,只要name相同,就分为一组,

输出结果如下:

四、第三种用法:

public static ienumerable<igrouping<tkey, telement>> groupby<tsource, tkey, telement>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, func<tsource, telement> elementselector);

官方释义:根据指定的键选择器函数对序列中的元素进行分组,并且通过使用指定的函数对每个组中的元素进行投影。

这个比第一种用法多了一个elementselector,第一种用法是对集合本身按照tkey分组,并将自己(tsource)添加到分组内,而当前的用法则可以选择自己想要添加到分组内的元素类型。

编写客户端实验代码如下:

var groups = personlist.groupby(p => p.gender, p=>p.name);
    foreach (var group in groups)
    {
      console.writeline(group.key.tostring());
      foreach(var name in group)
      {
        console.writeline($"\t{name}");
      }
    }

以上代码是按照p.gender进行分组,并将p.name作为组内的元素。

输出结果如下:

其等价的linq语句为:

var groups = from p in personlist
       group p.name by p.gender;

五、第四种用法:

public static ienumerable<tresult> groupby<tsource, tkey, tresult>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, func<tkey, ienumerable<tsource>, tresult> resultselector);

官方释义:根据指定的键选择器函数对序列中的元素进行分组,并且从每个组及其键中创建结果值。

这个跟之前的用法都不同,之前的用法都是将结果进行分组,并返回igrouping<tkey,tsource>对象,而当前用法则是返回自己定义的类型(tresult),在返回自己定义类型之前,将会传入两个参数,一个是tkey,为分组时指定的对象,另外一个则是ienumerable<tsource>,为分组后的内部对象集合。

编写客户端实验代码如下:

string getpersoninfo(string gender, ienumerable<person> persons)
      {
        string result = $"{gender}:\t";
        foreach (var p in persons)
        {
          result += $"{p.name},{p.age}\t";
        }
        return result;
      }
      var results = personlist.groupby(p => p.gender,(g, ps) => getpersoninfo(g,ps));
      foreach (var result in results)
      {
        console.writeline(result);
      }

getpersoninfo为局部方法,见于c#7.0及以上。

以上代码将分组后的内容(一个是tkey,为p.gender,另外一个是ienumerable<tsource>,为ienumerable<person>)作为字符串输出,因此,将返回的类型为字符串集合。

输出结果如下:

其等价的linq语句为:

      var results = from p in personlist
             group p by p.gender into pgroup
             select getpersoninfo(pgroup.key, pgroup);

六、第五种用法:

public static ienumerable<igrouping<tkey, telement>> groupby<tsource, tkey, telement>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, func<tsource, telement> elementselector, iequalitycomparer<tkey> comparer);

官方释义:根据键选择器函数对序列中的元素进行分组。通过使用比较器对键进行比较,并且通过使用指定的函数对每个组的元素进行投影。

与第三种用法基本相同,只是多了一个相等比较器,用于分组的依据。

使用第二种用法的personlist及personequalitycomparer,编写客户端实验代码如下:

var groups = personlist.groupby(p => p, p => new { p.age,p.gender },new personequalitycomparer());
      foreach (var group in groups)
      {
        console.writeline(group.key.tostring());
        foreach (var name in group)
        {
          console.writeline($"\t{name.age},{name.gender}");
        }
      }

以上代码的分组依据是person,personequalitycomparer则是作为person分组的比较器,每个组内为一个匿名类型集合。

输出结果如下:

七、第六种用法:

public static ienumerable<tresult> groupby<tsource, tkey, tresult>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, func<tkey, ienumerable<tsource>, tresult> resultselector, iequalitycomparer<tkey> comparer);

官方释义:根据指定的键选择器函数对序列中的元素进行分组,并且从每个组及其键中创建结果值。通过使用指定的比较器对键进行比较。

与第四种用法基本相同,只是多了一个相等比较器,用于分组的依据。

使用第二种用法的personlist及personequalitycomparer,编写客户端实验代码如下:

string getpersoninfo(person person, ienumerable<person> persons)
      {
        string result = $"{person.tostring()}:\t";
        foreach (var p in persons)
        {
          result += $"{p.age},{p.gender}\t";
        }
        return result;
      }
      var results = personlist.groupby(p => p, (p, ps) => getpersoninfo(p, ps),new personequalitycomparer());
      foreach (var result in results)
      {
        console.writeline(result);
      }

以上代码的分组依据是person,personequalitycomparer则是作为person分组的比较器,每个组内为一个person集合,并将返回类型为string的字符串输出。

输出结果如下:

八、第七种用法:

public static ienumerable<tresult> groupby<tsource, tkey, telement, tresult>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, func<tsource, telement> elementselector, func<tkey, ienumerable<telement>, tresult> resultselector);

官方释义:根据指定的键选择器函数对序列中的元素进行分组,并且从每个组及其键中创建结果值。通过使用指定的函数对每个组的元素进行投影。

与第四种方法很类似,只是对分组内的元素进行选择,原有为tsource,现改为telement。

编写客户端实验代码如下:

string getpersoninfo(string gender, ienumerable<string> names)
      {
        string result = $"{gender}:\t";
        foreach (var name in names)
        {
          result += $"{name}\t";
        }
        return result;
      }
      var results = personlist.groupby(p => p.gender, (p=>p.name) ,(g, ns) => getpersoninfo(g, ns));
      foreach (var result in results)
      {
        console.writeline(result);
      }

以上代码将使用gender分组,并将分组后的信息组合成一条字符串,并输出到控制台。

输出结果如下:

九、第八种用法:

public static ienumerable<tresult> groupby<tsource, tkey, telement, tresult>(this ienumerable<tsource> source, func<tsource, tkey> keyselector, func<tsource, telement> elementselector, func<tkey, ienumerable<telement>, tresult> resultselector, iequalitycomparer<tkey> comparer);

官方释义: 根据指定的键选择器函数对序列中的元素进行分组,并且从每个组及其键中创建结果值。通过使用指定的比较器对键值进行比较,并且通过使用指定的函数对每个组的元素进行投影。

与第七种用法基本相同,只是多了一个相等比较器,用于分组的依据。

使用第二种用法的personlist及personequalitycomparer,编写客户端实验代码如下:

var results = personlist.groupby(p => p, (p=>new { p.age,p.gender}),
        (p, ns) => 
        {
          string result = $"{p.tostring()}:\t";
          foreach (var n in ns)
          {
            result += $"{n.age},{p.gender}\t";
          }
          return result;
        },new personequalitycomparer());
      foreach (var result in results)
      {
        console.writeline(result);
      }

以上代码将使用person分组,使用person比较器作为分组的依据,并将分组后的信息组合成一条字符串,并输出到控制台。

输出结果如下:

以上就是c#在linq中使用groupby的详细内容,更多关于c#使用groupby的资料请关注移动技术网其它相关文章!

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网