.netcorecsharp 中级篇2-3
本节内容为linq及其拓展方法、linq中表达式树的使用
简介
语言集成查询(linq)是一系列直接将查询功能集成到c#语言的技术统称。数据查询历来都表示为简单的字符串,没有编译时类型检查或intellisense支持。此外,需要针对每种类型的数据源了解不同的查询语言:sql数据库、xml文档、各种web服务等。借助linq,查询成为了最高级的语言构造,就像类、方法和事件一样。可以使用语言关键字和熟悉的运算符针对强类型化对象集合编写查询。linq系列技术提供了针对对象(linqtoobjects)、关系数据库(linqtosql)和xml(linqtoxml)的一致查询体验。
对于编写查询的开发者来说,linq最明显的“语言集成”部分就是查询表达式。查询表达式采用声明性查询语法编写而成。使用查询语法,可以用最少的代码对数据源执行筛选、排序和分组操作。可使用相同的基本查询表达式模式来查询和转换sql数据库、ado.net数据集、xml文档和流以及.net集合中的数据。
在c#中可为以下对象编写linq查询:sqlserver数据库、xml文档、ado.net数据集以及支持ienumerable或泛型ienumerable 接口的任何对象集合。此外,第三方也为许多web服务和其他数据库实现提供了linq支持。
linq基本关键字介绍
- from …in:与foreach循环类似,将一个实现了ienumerable接口的数据进行迭代,通常是from 迭代变量 in 数据源的方式;
- select:将数据选中返回集合(ienumerable类型)
- where:后面接上相关限制语句,例如where a<3,如果直接返回,则会是一个iqueryable 数据。
- orderby,thenby:排序规则,后面接上排序的依据,例如orderby a.id thenby a.name
- groupby:分组依据,返回值回是一个类似于dictionary的数据(igouping)
-
join:联接操作在不同序列间创建关联,这些序列在数据源中未被显式模块化。例如,可通过执行联接来查找所有位置相同的客户和分销商。在 linq 中,join 子句始终作用于对象集合,而非直接作用于数据库表。
通常来说这几种是最为常用的linq关键字,还有一些关于linq查询的方法将在后面拓展方法中进行讲解linq基本操作
对于linq操作,其实非常类似于我们的sql语句操作,在以前ef并不完善的时候,linq to sql是一种非常好用的数据库操作语句。当然,linq也可以用于ienumerable及其派生类型的操作。
我们在这实际的进行一次类似于数据库操作的linq练习,案例是这样的,学生选课系统,那么存在学生与课程的关系,一个学生可以选多门课程,一个课程也可以有多个学生,对于这种关系,我们在代码中难以操作,因此引入中间表sc,记录学生的选课记录。假定我们不使用类似于ef中的集合去记录,只是单纯的使用代码将一切连接起来。
代码如下:
class student { public student(int id,string name) { studentid = id; name = name; } public int studentid { get; set; } public string name { get; set; } } class course { public int courseid { get; set; } public string cname { get; set; } public course(int sid,string name) { courseid = sid; cname = name; } } class sc { public int courseid { get; set; } public int studentid { get; set; } public sc(int cid,int sid) { courseid = cid; studentid = sid; } } class program { static void main(string[] args) { list<student> students = new list<student>() { new student(1,"mike"), new student(2,"jack"), new student(3,"david") }; list<course> courses = new list<course>() { new course(1,"cse"), new course(2,"cn"), new course(3,"swe") }; list<sc> scs = new list<sc>() { new sc(1,1), new sc(1,2), new sc(1,3), new sc(2,3), new sc(3,2) }; //筛选名称 var temp = from stu in students where stu.name == "jack" select stu; //级联多重查询,查询所有学生选课信息 var temp1 = from stu in students join scs in scs on stu.studentid equals scs.studentid join c in courses on scs.courseid equals c.courseid select new {stu.name,c.cname}; foreach(var t in temp1) { console.writeline("name:{0},course:{1}",t.name,t.cname); } } }
lambda表达式简要介绍
lambda表达式是一种看起来高大上的一种东西,本身我是想与委托一起进行讲解的,但是目前所接除到了我们的linq拓展方法,里面会涉及一些有关lambda表达式的操作,尤其是lambda表达式构造表达式树。
lambda表达式其实非常的简单,他是一种常见的语法糖,你可以将lambda表达式称为匿名函数,不过在linq中,他们常用作一种名为匿名委托的方式。我在本节中不会很详细的进行讲解我们的lambda表达式,只会告诉你如何在linq中使用。
在linq中,lambda表达式通常长这个样子
p => p.id == id && p.age > 5;
=>这个符号,就是lambda表达式的精髓,这个符号之前的p,是函数的返回值,当然也可能是没有的,不过在linq中,都是有的,因为我们需要一个匿名委托构造表达式树。而后面的所有,则是这个匿名函数的方法体。通常来说,linq中lambda表达式的方法体都会是一个类似于where判断型的语句,返回值通常是一个bool类型。
linq拓展方法
有了前面lambda表达式的一个简单的概念,我们就可以讲解一下linq中的拓展方法了,拓展方法提供了许多你使用linq关键字无法实现的操作。
常见的拓展方法有以下几种:
- where()
- firstordefault()
- join()
- groupby()
- orderby()
- max/min()
单从单词意思就能理解这些操作,我们使用的时候只需要使用类似p=>p.id即可。
例如:
students.where(p=>p.name == "jack").orderby(p=>p.studentid).firstordefault();
如果我的文章帮助了您,请您在github.netcoreguide项目帮我点一个star,在博客园中点一个关注和推荐。
github
bilibili主页
warrenryan’sblog