C#从委托、lambda表达式到linq总结
前言
本文总结学习C#必须知道的基础知识,委托、监视者模式、常用lambda表达式、linq查询,自定义扩展方法,他们之间有什么关系呢?匿名委托是如何演变成lambda表达式,lambda再如何导出linq语句的?
委托
用delegate关键字声明委托,引用MSDN上的一段内容:委托是一种引用方法的类型。一旦为委托分配了方法,委托将与该方法具有完全相同的行为。委托方法的调用可以像其他任何方法一样,具有参数和返回值。
using System;namespace ConsoleApplication1{//这个委托的名字是MyDel,委托的类型是MyDel//这个委托代表了一系列函数,这一些函数必须是没有返回值,没有参数的public delegate void MyDel();//这个委托代表了一系列函数,这一系列函数必须是没有返回值,参数为string类型public delegate void MyDel2(string name);//这个委托代表了一系列函数,这一系列函数必须是int类型的返回值,两个int类型的参数public delegate int MyDel3(int a,int b);class Program{static void Main(string[] args){//创建委托的2种方法//1.如果使用new关键字创建委托,则必须使用一个函数初始化这个委托对象MyDel2 my1 = new MyDel2(print);//2.如果不用new关键字,则可以直接赋值MyDel2 my2 = print2;//3.委托和他封装的方法具有相同的功能my1("ss");my2("ww");//4.既然委托代表了一系列函数,那么一个委托对象可以承接多个函数Console.WriteLine("委托对象承接多个函数");my1 += print2;my1("aa");//5.在承接的函数集中删减函数Console.WriteLine("委托对象删减函数");my1 -= print2;my1("bb");Console.ReadKey();}public static void print(string name) {Console.WriteLine("print-----------"+name);}public static void print2(string n) {Console.WriteLine("22222-----------"+n);}}}
监视者模式
监视者模式是在微软平台大量存在的一种模式,通俗一点它就是事件,事件就是监视者模式。比如生活中,你睡觉的时候委托同学上课叫醒你,这就是一个监视者模式,你是被监视者,是委托方,同学是被委托方,监视者。下面一个例子是:考试的时候自己委托父母监视自己,考的好的话让父母奖励,考差了则受罚的监视者模式:
using System;namespace ConsoleApplication1{public delegate void Del();class Program{static void Main(string[] args){//创建对象Student s = new Student();Parent p = new Parent();//s.p = () => { Console.WriteLine("超常发挥,耶");};//s.k = () => { Console.WriteLine("考差了,嘤嘤嘤"); };//学生对象的委托承接着家长对象的方法s.p += p.PPrice;s.k += p.KKit;//s开始执行考试s.Exam(70);Console.ReadKey();}}public class Student {public Del p = null;public Del k = null;//执行考试public void Exam(int f) {if (f < 60){k();}else {p();}}}public class Parent {public void PPrice() {Console.WriteLine("考的不错,奖励个馒头");}public void KKit() {Console.WriteLine("考的太差,面壁思过");}}}
匿名委托
匿名委托也是一种委托,只不过没有方法名,可以理解为用delegate代替了匿名委托的方法名,很多情况不必要创建方法,需要临时创建方法来调用时使用,下面例子很好的说明匿名委托的不同用法
using System;namespace ConsoleApplication1{public delegate void Delsing();public delegate int DelPlus(int a,int b);public delegate string DelString(string a,int b);class Program{static void Main(string[] args){//委托承接命名函数直接执行Delsing d = Sing;d();//匿名委托直接执行Delsing d1 = delegate() { Console.WriteLine("匿名委托"); };d1();//带参数且有返回值的匿名委托,执行后用write方法显示DelPlus d2 = delegate(int j, int k) { return j + k; };Console.WriteLine(d2(1,2));//带参数且有返回值的匿名委托,当做参数来传,然后调用函数实现功能;用定义的其他方法执行委托DelString d3=delegate(string a,int b){return a+b;};Test(d3,"1+1=",2);Console.ReadKey();}public static void Test(DelString del, string a, int b) {string str = del(a,b);Console.WriteLine(str);}public static void Sing() {Console.WriteLine("I'm singing");}}}
lambda表达式
lambda表达式其实就是匿名委托精简之后的形式,在参数和方法体中补上“=>”(称为goes to)来表示lambda表达式,下面是lambda表达式不同形式的总结
//先定义一个Del开头的委托public delegate void Del1();//匿名委托//转成lambda表达式时要去掉delegate 加上=>Del1 d1=delegate(){Console.WriteLine("ss");};d1=()=>{Console.WriteLine("ss");};//由于没有参数,那么()不能省略public delegate int Del2();Del2 d2=delegate(){return 1;};d2=()=>{return 1;};//如果是直接返回,换句话说就是没有业务逻辑处理,就是只有一条返回语句,可以把{}换成()同时去掉return关键字d2=()=>(1);//如果方法体中有业务逻辑,则必须使用{}d2=()=>{if(2>1){return 1;}else{return 2;}};//public delegate void Del3(string a);Del3 d3=delegate(string a){Console.WriteLine(a);};d3=(string a)=>{Console.WriteLine(a);};//可以把参数的类型去掉,因为系统会自动判断参数的类型,毕竟是把这个lambda赋值给了对应的委托d3=(a)=>{Console.WriteLine(a);};//若只有一个参数,则不需要()d3=a=>{Console.WriteLine(a);};//public delegate int Del4(int a,int a);Del4 d4=delegate(int a,int b){return a+b;};d4=(int a,int b)=>{return a+b;};d4=(a,b)=>{return a+b;};d4=(a,b)=>(a+b);d4=(a,b)=>a+b;d5=a=>a+1;
Linq语言集成化查询
linq不同于结构化查询语言(SQL)它不仅可以查询数据而且可以查询对象。
LINQ的官方中文名称为“.NET语言集成查询”,英文全称为“Language-Integrated Query”。它提供了类似于SQL语法的遍历,筛选与投影功能,LINQ不仅能完成对对象的查询,它可以透过DLINQ操作数据库,或是透过XLINQ控制XML。我们来比较一下两个不同方法求出数组中大于20的数据的例子:
using System;using System.Collections.Generic;namespace ConsoleApplication2{class Program{static void Main(){int[] a = { 2,23,25,32,5,64,52,30};//求出数组中大于20的数据List<int> list = new List<int>();foreach (int b in a) {if (b > 20) {list.Add(b);}}foreach (int c in list) {Console.WriteLine(c);}Console.ReadKey();}}}
2.用IEnumerable的Where扩展方法,通过lambda表达式,精简程序执行过程,需要导入命名空间:using System.Linq;
using System;using System.Collections.Generic;using System.Linq;namespace ConsoleApplication2{class Program{static void Main(){int[] a = { 2,23,25,32,5,64,52,30};//求出数组中大于20的数据//lambda表达式//var list = a.Where(p => { return p > 20; });//简化后 var可以接受任何类型var list = a.Where(p => p > 20);//Where 返回类型时IEnumerable//IEnumerable<int> list = a.Where(p=>p>20);foreach (int c in list) {Console.WriteLine(c);}Console.ReadKey();}}}
上面可以算是小的LINQ查询,但能算是真正的LINQ,我暂时理解为广义的LINQ。例子中(p=>p>20)其实是一个lambda表达式,是微软定义好的一个委托,从这个委托的特点我们知道它有一个参数,返回值是bool类型。数组肯定实现了IEnumerable接口,而Where是IEnumerable<(Of <T>)>成员的一个扩展方法,MSDN中定义为Where(Func<(Of<(UMP,Boolean)>)>)基于谓词筛选值序列。LINQ查询调用了微软定义好的扩展方法进行查询,下面我们插入一段扩展方法的内容,帮助理解上面LINQ的Where扩展方法。
扩展方法
通俗点说就是在不更改原来类的基础上,为类添加方法。需要注意的是:1.扩展方法必须写在静态类中;2.扩展方法必须是静态方法,虽然是静态方法,但这个扩展方法是为对象扩展的,只能由对象调用。它的定义是:
public static class 类名 {public static 返回值 方法名(this 要扩展的类型 对象名[,参数列表]) {}}
我们来做个扩展方法,为String类型扩展个获取文件类型的方法
using System;using System.Collections.Generic;using System.Linq;namespace ConsoleApplication2{public class Program{public static void Main(){//调用扩展方法获取文件类型string file = @"E:\FTPPUBLISH\学习资料\KindEditor\kindeditor-v4.0.3\examples\colorpicker.html";Console.WriteLine(file.GetFileType());string sss = "18.9.06.mp3";Console.WriteLine(sss.GetFileType());Console.ReadKey();}}public static class ExtendMethod{public static string GetFileType(this string str){string[] strs = str.Split('.');return strs[strs.Length - 1];}}}
理解了这些扩展方法,相信会有助于你理解上面LINQ查询时所用的Where扩展方法。下面我们继续来看LINQ查询,把上面的LINQ再做一步扩展:
用LINQ语句进行查询,并将结果降序分组排序的三种方式
using System;using System.Collections.Generic;using System.Linq;namespace ConsoleApplication2{public class Program{public static void Main(){List<Student> list = new List<Student>();list.Add(new Student(){Age=10,Name="Jack",Address="bj"});list.Add(new Student(){Age=67,Name="Mack",Address="郑州"});list.Add(new Student(){Age=23,Name="Dack",Address="USA"});list.Add(new Student(){Age=56,Name="Cack",Address="bj"});list.Add(new Student(){Age=8,Name="Eack",Address="郑州"});list.Add(new Student(){Age=34,Name="Hack",Address="bj"});list.Add(new Student(){Age=18,Name="小红",Address="USA"});Console.WriteLine("查询出集合中年龄大于45的学生");//查询出集合中年龄大于45的学生(完整形式,一般不这样写)//Func<Student, bool> f = p => p.Age > 45;//IEnumerable<Student> result = list.Where<Student>(f);//简写var result0 = list.Where<Student>(p=>p.Age>45);foreach (Student s in result0) {Console.WriteLine(s.Age+" "+s.Name);}Console.WriteLine("查询集合中年龄小于30,并按年龄降序排列,按城市分组");//查询集合中年龄小于30,并按年龄降序排列,按城市分组Console.WriteLine("____________________第一种方法____________________");IEnumerable<Student> result1 = list.Where(p => p.Age < 30).OrderByDescending(p => p.Age);IEnumerable<IGrouping<string, Student>> result11 = result1.GroupBy<Student, string>(p => p.Address);foreach (IGrouping<string, Student> gg in result11){foreach (Student s in gg){Console.WriteLine(s.Age + ";" + s.Name + ";" + s.Address);}}Console.WriteLine("____________________第二种方法____________________");var result2 = list.Where(p => p.Age < 30).OrderByDescending(p => p.Age).GroupBy(p=>p.Address);//第一次GetEnumerator()得到IEnumerator<IGrouping<string, Student>>var c = result2.GetEnumerator();while (c.MoveNext()) {//第二次GetEnumerator()得到IEnumerator<Student>var d = c.Current.GetEnumerator();while (d.MoveNext()) {//.Current获取集合中位于枚举数当前位置的元素Console.WriteLine(d.Current.Name+";"+d.Current.Address);}}Console.WriteLine("____________________第三种方法____________________");var result3 = from p in listwhere p.Age < 30orderby p.Age descendinggroup p by p.Address;foreach (var ss in result3) {foreach (var s in ss) {Console.WriteLine(s.Age + ";" + s.Name + ";" + s.Address);}}Console.ReadKey();}}public class Student {public int Age { get; set; }public string Name { get; set; }public string Address { get; set; }}}
最后我们做个小小的LINQ总结:LINQ语言集成化查询基础是泛型和lambda表达式,它的形式是:
from 元素 in 集合Where 元素条件orderby 元素.属性 ascending/descendinggroup 元素 by 元素.属性select 元素
和SQL查询类似,上面例子中表明如果使用了groupby语句,则不需要select。
参考资料
本文参考下文,并对文中的例子稍微做了修改
从委托、lambda表达式到linq的一些个人小总结
C#从委托、lambda表达式到linq总结的更多相关文章
- entity framework 新手入门篇(1.5)-lambda表达式与linq
在建立好了EF模型之后,先不着急使用它,在使用它之前,你还需要了解两个相关的技术,lambda表达式与linq. 作为微软C#语言中重要的语法糖-lambda表达式与LINQ,本质都是一个方法,以la ...
- c# in deep 之Lambda表达式于LINQ表达式结合后令人惊叹的简洁(2)
当Lambda表达式和LINQ一起使用时,我们会发现原本冗长的代码会变得如此简单.比如我们要打印0-10之间的奇数,让其从高到低排列并求其平方根,现在只用一行代码即可完成其集合的生成,直接上代码: v ...
- C# Lambda表达式和linq表达式 之 匿名对象查询接收
很多小伙伴都用过Lambda表达式和linq表达式,用起来也得心应手,但是有的小伙伴 对匿名对象的查询与接收比较迷茫,(没有定义相应的实体),怎么在where()里面进行 条件筛选呢?常规用法我就不说 ...
- .net学习之新语法学习(匿名类和匿名方法,扩展方法,系统内置委托,Lambda表达式和linq等)
1.自动属性 Auto-Implemented Properties 2.隐式类型 var var变量不能作为全局变量使用,因为不能在编译时确定类型 3.参数默认值 和 命名参数 4.对象初始化器 ...
- C#高级知识点概要(3) - 特性、自动属性、对象集合初始化器、扩展方法、Lambda表达式和Linq查询
1.特性(Attributes) 特性(Attributes),MSDN的定义是:公共语言运行时允许你添加类似关键字的描述声明,叫做attributes, 它对程序中的元素进行标注,如类型.字段.方法 ...
- .net Lambda表达式与Linq (LINQ TO object)
Lambda表达式,是用来写匿名方法的. 在委托用得比较多,因为委托是传递方法的. 定义几个委托: public delegate void DoNoThing();//无参无返回值 publ ...
- C#编程 委托 Lambda表达式和事件
委托 如果我们要把方法当做参数来传递的话,就要用到委托.简单来说委托是一个类型,这个类型可以赋值一个方法的引用. 声明委托 在C#中使用一个类分两个阶段,首选定义这个类,告诉编译器这个类由什么字段和方 ...
- C# 基础知识系列- 6 Lambda表达式和Linq简单介绍
前言 C#的lambda和Linq可以说是一大亮点,C#的Lambda无处不在,Linq在数据查询上也有着举足轻重的地位. 那么什么是Linq呢,Linq是 Language Intergrated ...
- 委托 lambda表达式浅显理解
方法不能跟变量一样当参数传递,怎么办,C#定义了委托,就可以把方法当变量一样传递了,为了简单,匿名方法传递,省得再声明方法了:再简单,lambda表达式传递,比匿名方法更直观. public dele ...
随机推荐
- CCF_ 201509-3_模板生成系统
又是一道考验细心和耐心的题,不知道哪里出问题了,一直只有90分 = =! #include<cstdio> #include<iostream> #include<cst ...
- 曹工说Spring Boot源码(16)-- Spring从xml文件里到底得到了什么(aop:config完整解析【上】)
写在前面的话 相关背景及资源: 曹工说Spring Boot源码(1)-- Bean Definition到底是什么,附spring思维导图分享 曹工说Spring Boot源码(2)-- Bean ...
- ZOJ 4067 Books (2018icpc青岛J) (贪心)
题意 给你一个长度为n的数组,代表每一个物品的价格.你有一个初始钱数\(x\),采用以下方法贪心: 从\(1\)到\(n\)扫一遍,如果\(x\)不比\(a[i]\)小,就买下它,买不起就跳过. 给你 ...
- spring中获取bean的方式
获取bean的方式 1.可以通过上下文的getBean方法 2.可以通过@Autowired注入 定义controller @RestController @RequestMapping(" ...
- Go语言实现:【剑指offer】不用加减乘除做加法
该题目来源于牛客网<剑指offer>专题. 第一步:相加各位的值,不算进位,得到010,二进制每位相加就相当于各位做异或操作,101^111. 第二步:计算进位值,得到1010,相当于各位 ...
- javascript json语句 与 js语句的互转
//var data = "weihexin" //var data = ["weihexin", 1] var data = {name:"weih ...
- 校招必看硬核干货:C++怎么学才能进大厂
目录 关于小猿 如何找资料 自我定位 岗位需求 学习路线及时间安排 资料获取方式 C++语言在历史舞台上出现了不短的时间,虽然一直面临着Python,Go等新语言的挑战,但它在基础架构和大型软件上的优 ...
- ELF文件之九——使用链接脚本-2个函数-data-bss-temp-call-debug信息-struct
main.c int enable; ; struct aaa { int membera; char memberb; }s_aaa; int main() { int temp; add(); d ...
- k8s系列---k8s认证及serviceaccount、RBAC
http://blog.itpub.net/28916011/viewspace-2215100/ 对作者文章有点改动 注意kubeadm创建的k8s集群里面的认证key是有有效期的,这是一个大坑!! ...
- expect 运行脚本文件 执行postgres数据库操作
#!/bin/bash /usr/bin/expect << EOF spawn /usr/local/pgsql/bin/.sh expect "*postgres:" ...