【.NET Core】Linq查询运算符(一)

简介: 笔记

一、概述


查询运算符是组成Linq模式的方法。这些方法中的大多数都作用于序列;其中序列指其类型实现IEnumberable接口或IQueryable接口的对象。标准查询运算符提供包括筛选、投影、集合、排序等查询功能。


查询运算符包含两组,一组作用于类型IEnumberable的对象,另一组作用于类型IQueryable的对象。


二、筛选数据


筛选是指将结果集限制为仅包含满足指定条件的元素的操作。它也称为选定内容。筛选数据方法包含OfType和Where。


string[] list=["A","AB","BC","CD","DE","EF"];
IEnumerable<string> query = from chars in list where chars.Contains("B") select chars;
foreach(string ch in query){
    Console.WriteLine(str);
}

运行结果:

AB
BC
1
2

三、投影运算


投影是指将对象转换为一种新形式的操作,该形式通常只包含那些将随后使用的属性。通过使用投影,可以构造从每个对象生成的新类型。可以投影属性,并对属性执行数学函数。


3.1 Select

string[] list={"A","AB","BC","CD","DE","EF"};
IEnumerable<string> query = from chars in list select chars.ToLower();
foreach(string ch in query){
    Console.WriteLine(str)
}

运行结果

a
ab
bc
cd
de
ef

3.2 SelectMany

使用多个from子句来投影字符串列表中每个字符串的每个字符转变为大写。

string[] list =  { "A|B|C|D","E|F|G|H"};
IEnumerable<string> query = from chars in list 
                            from ch in chars.Split('|')
                            select ch;
foreach (string ch in query)
{
  Console.WriteLine(ch)     
}

运行结果

A
B
C
D
E
F
G
H

3.3 Zip

Zip投影运算符有多个重载,所有Zip方法都处理两个或更多可能是异构类型的序列。前两个重载返回元组,具有来自给定序列的响应位置类型。

int[] index = { 1, 2, 3 };
string[] chars = { "A", "B", "C" };
IEnumerable<string> zip = chars.Zip(index, (n, w) => n + "=" + w);
foreach (var z in zip) 
{
   Console.WriteLine(z);
}

运行结果

A=1
B=2
C=3

zip操作生成的序列的长度永远不会长于最短序列。index和letters集合的长度不同,生成的序列将省略index集合中的最后一个元素,因为它没有任何要压缩的内容。


3.4 Select 与 SelectMany

Select与SelectMany的工作都是依据源生成一个或多个结构值。Select为每个源值生成一个结构值。因此,总体结构是一个与源集合具有相同元素数目的集合。 与之相反,SelectMany 生成单个总体结果,其中包含来自每个源值的串联子集合。 作为参数传递到 SelectMany 的转换函数必须为每个源值返回一个可枚举值序列。 然后,SelectMany 串联这些可枚举序列,以创建一个大的序列。


四、Set(设置)运算


LINQ 中的集运算是指根据相同或不同集合(或集)中是否存在等效元素来生成结果集的查询运算。


4.1 Distinct或DistinctBy

删除集合中的重复元素

int[] index = { 1, 2, 3 ,4};
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> zip = from ch in chars.Distinct() select ch;
foreach (var z in zip) 
{
     Console.WriteLine(z);
}

运行结果

A
B
C
D

在新.NET Framework中可以使用DistinctBy替代Distinct,DistinctBy采用keySelector。keySelector用作源类型的比较鉴别器。


4.2 Except或ExceptBy

Except返回的序列只包含位于第一个输入序列但不位于第二个输入序列的元素。

string[] listA = { "A", "B", "AC", "BC", "CD" };
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> listC = from ch in chars.Except(listA) select ch;
foreach (var z in listC) 
{
    Debug.WriteLine(z);
}

运行结果

C
D


ExceptBy 方法是 Except 的替代方法,它采用可能是异构类型的两个序列和一个 keySelector。 keySelector 与第二个集合的类型相同,用作源类型的比较鉴别器。


4.3 Intersect或IntersectBy

返回序列包含两个输入序列共有的元素。

string[] listA = { "A", "B", "AC", "BC", "CD" };
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> listC = from ch in chars.Intersect(listA) select ch;
foreach (var z in listC) 
{
    Debug.WriteLine(z);
}

运行结果

A
B


IntersectBy方法是Intersect的替代方法,它采用可能是异构类型的两个序列和一个keySelector。keySelector用作第二个集合类型的比较鉴别器。


4.4 Union或UnionBy

两个字符串序列执行的联合操作。返回的序列包含两个输入序列的唯一元素。

string[] listA = { "A", "B", "AC", "BC", "CD" };
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> listC = from ch in chars.Union(listA) select ch;
foreach (var z in listC) 
{
     Debug.WriteLine(z);
}

运行结果

A
B
C
D
AC
BC
CD


五、对数据进行排序


排序操作基于一个或多个属性对序列的元素进行排序。 第一个排序条件对元素执行主要排序。 通过指定第二个排序条件,您可以对每个主要排序组内的元素进行排序。


5.1 OrderBy

结果集按升序对值排序。

string[] listA = { "A", "B", "AC", "BC", "CD" };
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> listC = from ch in chars.Union(listA) orderby ch select ch;
foreach (var z in listC) 
{
   Debug.WriteLine(z);
}

运行结果

A
AC
B
BC
C
CD
D

5.2 OrderByDescending

按降序对值排序。C#查询表达式语法orderby… descending


string[] listA = { "A", "B", "AC", "BC", "CD" };
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> listC = from ch in chars.Union(listA) 
                            orderby ch descending 
                            select ch;
foreach (var z in listC) 
{
     Debug.WriteLine(z);
}

运行结果

D
CD
C
BC
B
AC
A

5.3 ThenBy

按升序执行次要排序。orderby...,...


string[] listA = { "A", "B", "AC", "BC", "CD" };
string[] chars = { "A", "B", "C" , "B", "C" ,"D"};
IEnumerable<string> listC = from ch in chars.Union(listA) orderby ch,ch.Length select ch;
foreach (var z in listC) 
{
   Debug.WriteLine(z);
}

运行结果

A
AC
B
BC
C
CD
D


5.4 ThenByDescending

按降序执行次要顺序。C#查询表达式语法orderby…,… descending。


string[] listA = { "AB", "DC", "ED", "FH", "Z" };
IEnumerable<string> listC = from ch in listA
                            orderby ch descending,ch.Length  descending
                            select ch;
foreach (var z in listC) 
{
    Debug.WriteLine(z);
}

运行结果

Z
FH
ED
DC
AB


5.5 Reverse()

反转集合中元素的顺序

string[] listA = { "AB", "DC", "ED", "FH", "Z" };
IEnumerable<string> listC = listA.Reverse();
foreach (var z in listC) 
{
     Debug.WriteLine(z);
}

运行结果

Z
FH
ED
DC
AB


六、限定符运算


限定符运算返回一个Boolean值,该值指示序列中是否有一些元素满足条件或是否所有元素都满足条件。


6.1 All

全部确定是否序列中的所有元素都满足条件


class Marketing
{
   public string Name{get;set;}
   public string[] Items{get;set;}
}
public static void Main(string[] args)
{
   List<Marketing> markets=
   {
       new Market { Name = "Emily's", Items = {"kiwi", "cheery", "banana"} },
       new Market { Name = "Kim's", Items = {"melon", "mango", "olive"} },
       new Market { Name = "Adam's", Items = {"kiwi", "apple", "orange"} },
   } 
   var names = from marker in markets
               where marker.Items.All(item=>item.Length == 5)
               select Name;
    foreach(string name in namse)
    {
        Debug.WriteLine(name);
    }
}

6.2 Any

使用Any检查所有字符串是否以“o”开头。

class Market
{
    public string Name{get;set;}
    public string[] Items{get;set;}
}
public static void Main(string[] args)
{
    List<Market> markets=
    {
       new Market { Name = "Emily's", Items = {"kiwi", "cheery", "banana"}},
       new Market { Name = "Kim's", Items = {"melon", "mango", "olive"} },
       new Market { Name = "Adam's", Items = {"kiwi", "apple", "orange"} },  
    }
    IEnumerable<string> names = from market in markets
                                where market.Items.Any(item=>item.StartsWith("o"))
                                select market.Name;
    foreach(string name in names)
    {
        Debug.WriteLine(name);
    }
}

6.2 Contains

Contains检查所有数组是否具有特定元素。

class Market
{
    public string Name{get;set;}
    public string[] Items{get;set;}
}
public static void Main(string[] args)
{
    List<Market> markets=
    {
       new Market { Name = "Emily's", Items = {"kiwi", "cheery", "banana"}},
       new Market { Name = "Kim's", Items = {"melon", "mango", "olive"} },
       new Market { Name = "Adam's", Items = {"kiwi", "apple", "orange"} },  
    }
    IEnumerable<string> names = from market in markets
                                where market.Items.Contains("kiwi")
                                select market.Name;
    foreach(string name in names)
    {
        Debug.WriteLine(name);
    }
}


七、数据分区


Linq中的数据分区是指将输入序列划分为两个部分的操作,无需重新排列元素,然后返回其中一个部分。


7.1 Skip

Skip跳过序列中指定位置之前的元素。

string[] items ={"A", "B","C","D","E","F","G","H","I"};
var item= items.Skip(5);
foreach (var it in item) 
{
    Console.WriteLine(it);
}

运行结果

F
G
H
I


7.2 SkipWhile

Skip指定跳过的数目,SkipWhile指定的跳过条件,而不是元素数。

string[] texts = new string[] { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
IEnumerable<string> skippedTexts = texts.SkipWhile(value => value.EndsWith("n"));
foreach (var it in skippedTexts) 
{
   Console.WriteLine(it);
}

运行结果

Tue
Wed
Thu
Fri
Sat

7.3 Take

获取序列中指定位置之前的元素

string[] texts = new string[] { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
// 删除尾字母为n的
IEnumerable<string> skippedTexts = texts.Take(3);
foreach (var it in skippedTexts) 
{
    Console.WriteLine(it);
}

运行结果

Sun
Mon
Tue


7.4 TakeWhile

TakeWhile操作符用于从输入序列中返回指定数量且满足一定条件的元素。


当TakeWhile操作符被调用时,会将source序列中的每一个元素顺序传递给委托predicate,只有哪些使用得predicate返回值为true的元素才会被添加到结果序列中,要特别注意是,当TakeWhile操作符在查找过程中,遇到第一个返回false的元素就会立即停止执行,跳出,无论后面还有没有符合条件的元素,即使后面有符合条件的元素也不会获取。对于第二个扩展方法,委托里面含有一个int类型的参数。

string[] lists = ["a", "e", "i", "o", "u"];
var takeList = lists.TakeWhile(x => x != "i");
foreach (var al in takeList) 
{
    Console.WriteLine(al);
}

运行结果

a
e

7.5 Chunk

chunk 该方法将序列的元素拆分为指定大小的区块

string[] lists = {"公孙胜","鲁智深","林冲","吴用","李逵","宋江","武松" };
var nameList = lists.Chunk(3);
foreach (var names in nameList) 
{
    foreach (var name in names) 
    {
        Trace.WriteLine(name);
    }
    Trace.WriteLine("****************");
}

运行结果

公孙胜
鲁智深
林冲
****************
吴用
李逵
宋江
****************
武松
****************


目录
相关文章
|
15天前
|
开发框架 .NET 开发者
简化 ASP.NET Core 依赖注入(DI)注册-Scrutor
Scrutor 是一个简化 ASP.NET Core 应用程序中依赖注入(DI)注册过程的开源库,支持自动扫描和注册服务。通过简单的配置,开发者可以轻松地从指定程序集中筛选、注册服务,并设置其生命周期,同时支持服务装饰等高级功能。适用于大型项目,提高代码的可维护性和简洁性。仓库地址:&lt;https://github.com/khellang/Scrutor&gt;
37 5
|
2月前
|
存储 开发框架 JSON
ASP.NET Core OData 9 正式发布
【10月更文挑战第8天】Microsoft 在 2024 年 8 月 30 日宣布推出 ASP.NET Core OData 9,此版本与 .NET 8 的 OData 库保持一致,改进了数据编码以符合 OData 规范,并放弃了对旧版 .NET Framework 的支持,仅支持 .NET 8 及更高版本。新版本引入了更快的 JSON 编写器 `System.Text.UTF8JsonWriter`,优化了内存使用和序列化速度。
|
1月前
|
开发框架 .NET C#
在 ASP.NET Core 中创建 gRPC 客户端和服务器
本文介绍了如何使用 gRPC 框架搭建一个简单的“Hello World”示例。首先创建了一个名为 GrpcDemo 的解决方案,其中包含一个 gRPC 服务端项目 GrpcServer 和一个客户端项目 GrpcClient。服务端通过定义 `greeter.proto` 文件中的服务和消息类型,实现了一个简单的问候服务 `GreeterService`。客户端则通过 gRPC 客户端库连接到服务端并调用其 `SayHello` 方法,展示了 gRPC 在 C# 中的基本使用方法。
42 5
在 ASP.NET Core 中创建 gRPC 客户端和服务器
|
23天前
|
开发框架 缓存 .NET
GraphQL 与 ASP.NET Core 集成:从入门到精通
本文详细介绍了如何在ASP.NET Core中集成GraphQL,包括安装必要的NuGet包、创建GraphQL Schema、配置GraphQL服务等步骤。同时,文章还探讨了常见问题及其解决方法,如处理复杂查询、错误处理、性能优化和实现认证授权等,旨在帮助开发者构建灵活且高效的API。
26 3
|
1月前
|
XML 开发框架 .NET
.NET 9 中 LINQ 新增功能实操
.NET 9 中 LINQ 新增功能实操
|
3月前
|
开发框架 监控 前端开发
在 ASP.NET Core Web API 中使用操作筛选器统一处理通用操作
【9月更文挑战第27天】操作筛选器是ASP.NET Core MVC和Web API中的一种过滤器,可在操作方法执行前后运行代码,适用于日志记录、性能监控和验证等场景。通过实现`IActionFilter`接口的`OnActionExecuting`和`OnActionExecuted`方法,可以统一处理日志、验证及异常。创建并注册自定义筛选器类,能提升代码的可维护性和复用性。
|
2月前
|
开发框架 JavaScript 前端开发
一个适用于 ASP.NET Core 的轻量级插件框架
一个适用于 ASP.NET Core 的轻量级插件框架
|
2月前
|
开发框架 .NET 开发工具
.NET 9 中 LINQ 新增的功能
.NET 9 中 LINQ 新增的功能
|
3月前
|
开发框架 前端开发 JavaScript
ASP.NET MVC 教程
ASP.NET 是一个使用 HTML、CSS、JavaScript 和服务器脚本创建网页和网站的开发框架。
51 7
|
3月前
|
存储 开发框架 前端开发
ASP.NET MVC 迅速集成 SignalR
ASP.NET MVC 迅速集成 SignalR
82 0