LINQ :最终统治了​所有的语言!

LINQ:最终统治了所有的语言!

让我们看看LINQ如何彻底改变了.NET中访问数据的方式

.NET与其他技术栈的不同之处之一绝对是LINQ,它是Language Integrated Query的首字母缩写。实际上,它是随.NET Framework 3.5和Visual Studio 2008引入的,它是第一个独立于体系结构并集成在C#和Visual Basic语言中的框架。

借助LINQ,我们可以使用独立于各种源的单个编程模型来查询和操作数据。为了更好地理解它是什么,我们必须一步步的看下他的发展历程。

在C#的第一个版本中,我们必须使用for或foreach循环来遍历一个集合,正如我们所知,该集合实现IEnumerable接口,例如,在其中找到一个特定的对象。以下代码返回公司年龄在19至36岁(20至35岁)之间的所有客户:

class Customer
{ public int CustomerID { get; set; } public String CustomerName { get; set; } public int Age { get; set; } 
} class Program 
{ static void Main(string[] args) { Customer[] customerArray = { new Customer() { CustomerID = 1, CustomerName = "Joy", Age = 22 }, new Customer() { CustomerID = 2, CustomerName = "Bob", Age = 45 }, new Customer() { CustomerID = 3, CustomerName = "Curt", Age = 25 },};Customer[] customers = new Customer[10]; int i = 0; foreach (Customer cst in customerArray) { if (cst.Age > 19 && cst.Age < 36) { customers[i] = cst; i++; }} } 
}

有什么不同的方法?让我们尝试从“委托”的概念开始逐步进行开发。delegate和返回的类型的方法的引用类型。它“委托”它旨在执行代码的方法,我们可以用这种方式声明它:

public delegate bool Operations(int number);

此委托可以指向所有接受输入整数并返回布尔值的方法。例如,假设在CustomerOperations类中有一个方法:

public bool CustomerAgeRangeCheck(int number)
{return number > 19 && number < 36; 
}

我们可以注册一个或多个将在执行委托时执行的方法:

Operations op = new Operations(CustomerOperations.CustomerAgeRangeCheck);

或者简单地:

 Operations op = CustomerOperations.CustomerAgeRangeCheck; 

因此,我们可以使用委托,在这种情况下,它将返回true:

 op(22); 

委托用于将方法作为参数传递给其他方法:事件处理程序和回调是通过委托调用的方法的示例。

C#2.0引入了匿名委托,您现在可以使用匿名方法来声明和初始化委托。例如,我们可以这样写:

delegate bool CustomerFilters(Customer customer);class CustomerOperations
{public static Customer[] FindWhere(Customer[] customers, CustomerFilters  customerFiltersDelegate){int i = 0;Customer[] result = new Customer[6];foreach (Customer customer in customers)if (customerFiltersDelegate(customer)){result[i] = customer;i++;}return result;}
}class Program
{static void Main(string[] args){Customer[] customers = {new Customer() { CustomerID = 1, CustomerName = "Joy", Age = 22 },new Customer() { CustomerID = 2, CustomerName = "Bob", Age = 45 },new Customer() { CustomerID = 3, CustomerName = "Curt", Age = 25 },};Customer[] filteredCustomersAge = CustomerOperations.FindWhere(customers, delegate (Customer customer)  //Using anonimous delegate{return customer.Age > 19 && customer.Age < 36;});}
}

使用C#2.0,我们的优势是可以使用匿名委托在不同条件下进行搜索,而无需使用for或foreach循环。例如,我们可以使用上一个示例中相同的委托函数来查找“ CustomerID”为3或名称为“ Bob”的客户:

Customer[] filteredCustomersId = CustomerOperations.FindWhere(customers, delegate (Customer customer)  {return customer.CustomerID == 3;});Customer[] filteredCustomersName = CustomerOperations.FindWhere(customers, delegate (Customer customer)  {return customer.CustomerName == "Bob";});

随着C#的发展,从3.x版本开始,Microsoft团队引入了新功能,使代码更加紧凑和易读。这些直接支持LINQ来查询不同类型的数据源并获得产生单个指令的元素。

这些功能是:

-在VAR结构,一个隐式类型的局部变量。它是强类型化的,因为已经声明了类型本身,但是由编译器根据分配给它的值使用类型推断来确定类型。以下两个语句在功能上等效:

var customerAge = 30; // Implicitly typed.
int customerAge = 30; // Explicitly typed.

-使用对象初始化程序,您可以在对象创建期间将值分配给对象的全部或某些属性,而无需在分配指令行之后调用构造函数。

Customer customer = new Customer { Age = 30, CustomerName = "Adolfo" };

与以下代码不同,在前一种情况下,所有内容都被视为单个操作。

Customer customer = new Customer();
customer.Age = 30; 
customer.CustomerName = "Adolfo";

匿名类型,由编译器构建的只读类型,只有编译器知道它。但是,如果程序集中的两个或多个匿名对象初始化程序具有相同顺序的属性序列,并且具有相同的名称和类型,则编译器会将这些对象视为相同类型的实例。
匿名类型是将查询结果中的一组属性临时分组的好方法,而不必定义单独的命名类型。 

var customer = new { YearsOfFidelity = 10, Name = "Francesco"}; 

-扩展方法,使您可以将方法“添加”到现有类型,而无需创建新的派生类型,重新编译或修改原始类型。扩展方法是静态方法,但由于引入了语法糖,因此被称为,因为它们是扩展类型上的实例方法。

public static class StringExtensionMethods
{public static string ReverseString(this string input){if (string.IsNullOrEmpty(input)) return "";return new string(input.ToCharArray().Reverse().ToArray());}
}

扩展方法必须在静态类中定义。第一个参数表示要扩展的类型,并且必须以关键字this开头,其他参数则不需要它。

Console.WriteLine("Hello".ReverseString());   //olleH

请注意,在方法调用中不得指定第一个参数,该参数以this修饰符开头。

Lambda表达式,可作为可变的或作为在一方法调用中的参数被传递匿名函数。

customer => customer.Age > 19 && customer.Age < 36;

=>运算符称为lambda运算符,而customer是函数的输入参数。lambda运算符右侧的部分代表函数的主体及其返回的值,在这种情况下为布尔值。

在LINQ的引入中,我们终于有了C#3.5版本。

简而言之,我们可以说LINQ是IEnumerable 和IQueryable 接口的扩展方法库,它使我们能够执行各种操作,如过滤,进行投影,聚合和排序。

我们有几种可用的LINQ实现:

•LINQ到对象(内存中对象集合)•LINQ到实体(实体框架)•LINQ to SQL(SQL数据库)•LINQ to XML(XML文档)•LINQ到数据集(ADO.Net数据集)•通过实现IQueryable接口(其他数据源)

在前面的示例中,数组用作数据源,因此隐式支持通用接口IEnumerable <T <。支持IEnumerable或其派生接口的类型(例如通用IQueryable 接口)称为可查询类型,使我们可以直接执行LINQ查询。如果数据源尚未以可查询类型存储在内存中,则LINQ提供程序必须将其表示为可查询类型。

正如我们所说,LINQ查询主要基于.NET Framework 2.0版中引入的通用类型。这意味着,例如,如果尝试将Customer对象添加到List对象,则在编译时将生成错误。使用通用集合很容易,因为不需要在运行时强制转换类型。
如果愿意,可以使用前面提到的var关键字来避免通用语法,在下面的示例中,该关键字要求编译器通过检查from子句中指定的数据源来推断查询变量的类型。
因此,让我们看看如何能达到同样的效果,在前面的代码中我们获得了使用匿名委托,使用LINQ到对象查询,该变种构造和lambda表达式:

var filteredCustomersAge = customers.Where(c => c.Age > 19 && c.Age < 36);

这种语法称为方法语法。
在下一个示例中,我们将使用查询语法(Query Syntax),该语法是为那些已经了解SQL语言并且因此会喜欢这种方法的人引入的:

var filteredCustomersAge =from customer in customerswhere customer.Age > 19 && customer.Age < 36select customer;

查询语法和方法语法在语义上是相同的,许多人发现查询的语法更简单,更易于阅读 在查询语法中,LINQ查询运算符在编译时转换为对相关LINQ扩展方法的调用。

在下一篇文章中,我们将继续讨论LINQ!
我们将讨论IQueryable 接口,其相关的LINQ扩展方法以及与IEnumerable 接口的区别。
与远程数据库一样,我们还将看到LINQ与内存外集合的数据源一起使用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/308805.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[SpringBoot2]文件上传_单文件与多文件上传的使用

<form role"form" th:action"{/upload}" method"post" enctype"multipart/form-data"><div class"form-group"><label for"exampleInputEmail1">邮箱</label><input type"email&…

使用Docker运行SQL Server

现在.net core已经跨平台了&#xff0c;大家也都用上了linux用上了docker。跟.net经常配套使用的SQL SERVER以前一直是windows only&#xff0c;但是从SQL Server 2017开始已经支持运行在docker上&#xff0c;也就说现在SQL Serer已经可以运行在linux下了。下面在Ubuntu 16.4上…

[mybatis]映射文件_select_resultMap_关联查询_association分步查询延迟加载

association分步查询 场景一 查询Employee的同时查询员工对应的部门 EmployeeDepartment 一个员工有与之对应的部门信息 Employee表: Department表&#xff1a; public interface DepartmentMapper {public Department getDeptById(Integer id);}public interface EmployeeMap…

C#中你想象的Task,很简单?

【导读】网上关于Task的文章如数家珍&#xff0c;不过有一部分并未谈到一个根本的问题&#xff0c;所创建的Task一定在线程池上运行&#xff1f;如何合理的使用Task&#xff1f;这里并不会去重新讲解每一个APi的使用&#xff0c;没有任何意义&#xff0c;这属于包括我在内的各位…

【复杂系统迁移 .NET Core平台系列】之应用发布与部署

源宝导读&#xff1a;微软跨平台技术框架—.NET Core已经日趋成熟&#xff0c;已经具备了支撑大型系统稳定运行的条件。本文将介绍明源云ERP平台从.NET Framework向.NET Core迁移过程中的实践经验。一、背景随着ERP的产品线越来越多&#xff0c;业务关联也日益复杂&#xff0c;…

不懂数据库索引原理?因为你心里没有一点B树

什么是B树&#xff1f;B树是一种数据结构它按排序顺序在其节点中存储数据&#xff0c;B树存储数据使得每个节点按升序包含密钥&#xff0c;这些键中的每一个都有两个对另外两个子节点的引用&#xff0c;Te左侧子节点键小于当前键右侧子节点键多于当前键&#xff0c;如果单个节点…

[mybatis]缓存_二级缓存使用细节

二级缓存 开启全局二级缓存 <setting name"cacheEnabled" value"true"/>去mapper.xml中配置使用二级缓存 <cache></cache><cache eviction"FIFO" flushInterval"60000" readOnly"false" size"1…

5分钟快速接入钉钉实现钉钉考勤

一、前言由于今年疫情影响&#xff0c;假期的无限延长让大家都不得不进行线上办公&#xff0c;说到线上办公就毫无疑问&#xff0c;钉钉是这个疫情假期最大的赢家&#xff0c;APP的火热程度以及下载量甚至压过了微信&#xff0c;跃居App store免费排行榜第1名的位置。最早我们知…

[mybatis]缓存机制介绍_一级缓存二级缓存

两级缓存&#xff1a; 一级缓存&#xff1a;&#xff08;本地缓存&#xff09;&#xff1a;sqlSession级别的缓存。一级缓存是一直开启的&#xff1b;sqlSession级别的一个Map ​ 与数据库同一次会话期间查询到的数据会放在本地缓存中。 以后如果需要获取相同的数据&#xff0c…

使用Azure Application Insignhts监控ASP.NET Core应用程序

Application Insignhts是微软开发的一套监控程序。他可以对线上的应用程序进行全方位的监控&#xff0c;比如监控每秒的请求数&#xff0c;失败的请求&#xff0c;追踪异常&#xff0c;对每个请求进行监控&#xff0c;从http的耗时&#xff0c;到SQL查询的耗时&#xff0c;完完…

[mybatis]逆向工程MGB基本编写

逆向工程 项目结构 依赖 <dependency><groupId>org.mybatis.generator</groupId><artifactId>mybatis-generator-core</artifactId><version>1.3.7</version> </dependency>mgb.xml <?xml version"1.0" en…

[SpringBoot2]HelloWorld

导入依赖 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.3.4.RELEASE</version></parent><dependencies><dependency><groupId>or…

Linux内核:容器底层cgroup如何使用

在前面的文章中&#xff0c;我们探讨了容器底层 cgroup 的数据结构与代码实现&#xff0c;本期是 cgroup 系列的最后一篇文章&#xff0c;我们将继续探讨在 mount 成功后&#xff0c;我们如何使用 cgroup 来实现进程限制。在 mount 成功后&#xff0c;cgroup_root 已经存在了&a…

[SpringBoot2]容器功能_底层注解配置绑定_@Configuration@Import@Conditional@ImportResource

Configuration&Bean 告诉SpringBoot这是一个配置类配置文件 #############################Configuration使用示例###################################################### /*** 1、配置类里面使用Bean标注在方法上给容器注册组件&#xff0c;默认也是单实例的* 2、配置…

.NET Core微服务开发服务间调用篇-GRPC

在单体应用中&#xff0c;相互调用都是在一个进程内部调用&#xff0c;也就是说调用发生在本机内部&#xff0c;因此也被叫做本地方法调用&#xff1b;在微服务中&#xff0c;服务之间调用就变得比较复杂&#xff0c;需要跨网络调用&#xff0c;他们之间的调用相对于与本地方法…

基于.NetCore3.1搭建项目系列 —— 认证授权方案之Swagger加锁

1开始在之前的使用Swagger做Api文档中&#xff0c;我们已经使用Swagger进行开发接口文档&#xff0c;以及更加方便的使用。这一转换&#xff0c;让更多的接口可以以通俗易懂的方式展现给开发人员。而在后续的内容中&#xff0c;为了对api资源的保护&#xff0c;我们引入了认证授…

今晚8点,dotnet课堂全新起航,张善友/陈计节/刘腾飞我们一起来聊聊abp的故事...

直播主题&#xff1a;我们和Abp的故事直播嘉宾&#xff1a;张善友&#xff0c;陈计节&#xff0c;刘腾飞直播话题张善友&#xff1a;我是如何使用Abp的刘腾飞&#xff1a;利用Abp的模块化解决单体和分布式混合架构陈计节&#xff1a;Abp开源项目的DevOps实践Abp VNext 处于被低…

你知道技术委员会吗?嗯,一个既重要却又鸡肋的神秘组织

这是头哥侃码的第209篇原创前几天&#xff0c;在某群里有个有关 “技术委员会” 的话题讨论的挺火。很多时候&#xff0c;这种争论是不可能有结果的&#xff0c;因为每个人都会从自己的立场和三观出发&#xff0c;公说公有理&#xff0c;婆说婆有理&#xff0c;一般除了搞得唾沫…