知识集锦
一、lambda表达式介绍
无参数
() =>{return "1";};
等同于
string getnum(){ return "1"; }
有两个参数
(p1, p2) =>{ return p1*p2;};
等同于
int mul(p1, p2) { return p1*p2;};
lambda表达式可以捕获外部变量,并在其主体中使用它们。这些变量被称为捕获变量,它们会在lambda表达式创建时被复制,并在lambda表达式执行时被更新。例如:
int factor = 2;Func<int, int> multiplier = n => n * factor; //==》 当结构体只有一句的时候可以省略 return ,多条语句不可省略Console.WriteLine(multiplier(3)); //输出6factor = 3;Console.WriteLine(multiplier(3)); //输出9
二、lambda演进过程
//声明委托类型
public delegate void SayHandler(string name, int age);
//原始形态
SayHandler handler1=new SayHandler( delegate(string name, int age){ Console.WriteLine("你好"); }
);
//进化1 直接去掉:new SayHandler( )
SayHandler handler2=delegate(string name, int age)
{ Console.WriteLine("你好");
}
//进化2 delegate去掉后,换上了一个符号“=>”,读作:goes to
SayHandler handler3=(string name, int age)=>{ Console.WriteLine("你好"); }
//进化3 因为与委托类型是一一对应关系,所以直接去掉参数类型名称
SayHandler handler4=(name, age)=>{ Console.WriteLine("你好"); }
//进化4 匿名方法只有一行代码,则花括号{} 可省略
//注:若参数列表中只有一个参数,则圆括号() 也可省略
SayHandler handler5=(name, age)=> Console.WriteLine("你好");
三、lambda表达式的5中高频使用方法
第一种 使用lambda表达式作为委托或事件处理程序
1.1 委托是一种引用类型,它可以封装一个具有特定参数和返回值类型的方法。委托可以用来实现回调函数、事件处理程序、匿名方法等功能。lambda表达式可以直接表示为委托的代码,而不需要单独定义一个方法或一个委托类型。这样可以简化委托的创建和使用,提高代码的可读性和灵活性。
//定义一个委托类型
delegate int CalculateHandler(int x, int y);//创建一个lambda表达式,并赋值给一个委托变量
CalculateHandler sumHandler = (x, y) => x + y;//调用委托变量,执行lambda表达式的主体
Console.WriteLine(sumHandler(1, 2)); //输出3//使用系统预定义的泛型委托类型
Func<int, int, int> multiplyHandler = (x, y) => x * y;//调用委托变量,执行lambda表达式的主体
Console.WriteLine(multiplyHandler(2, 3)); //输出6//直接将lambda表达式作为参数传递给其他方法
List<int> numbers = new List<int>() { 1, 2, 3, 4, 5 };
numbers.ForEach(n => Console.WriteLine(n * n)); //输出1 4 9 16 25
1.2 事件是一种特殊的委托,它可以在某个对象发生特定行为时通知其他对象。事件处理程序是一种特殊的方法,它可以在事件被触发时执行一些操作。lambda表达式可以直接表示为事件处理程序的代码,而不需要单独定义一个方法。这样可以简化事件处理程序的创建和使用,提高代码的可读性和灵活性
//定义一个按钮类
class Button
{//定义一个点击事件public event EventHandler Click;//定义一个触发点击事件的方法public void OnClick(){//如果有订阅了该事件的处理程序,则调用它们Click?.Invoke(this, EventArgs.Empty);}
}//创建一个按钮对象
Button button = new Button();//创建一个lambda表达式,并添加到点击事件的订阅列表中
button.Click += (sender, e) =>
{Console.WriteLine("Button clicked");
};//触发点击事件,执行lambda表达式的主体
button.OnClick(); //输出Button clicked
第二种:使用lambda表达式作为LINQ查询的谓词或选择器
LINQ是一种语言集成查询,它可以用来对各种数据源进行查询、筛选、排序、分组等操作。LINQ查询可以使用查询表达式或方法语法来编写,其中方法语法可以使用lambda表达式来指定查询的条件或投影。lambda表达式可以直接表示为谓词或选择器的代码,而不需要单独定义一个方法。这样可以简化LINQ查询的创建和使用,提高代码的可读性和灵活性。
//定义一个数据源
List<int> numbers = new List<int>() { 1, 2, 3, 4, 5 };//创建一个lambda表达式,并作为参数传递给Where方法,筛选出偶数var evenNumbers = numbers.Where(n => n % 2 == 0);//遍历LINQ查询结果,执行lambda表达式的主体
foreach (var n in evenNumbers)
{Console.WriteLine(n); //输出2 4
}//创建一个lambda表达式,并作为参数传递给Select方法,将每个数乘以10
var multipliedNumbers = numbers.Select(n => n * 10);//遍历LINQ查询结果,执行lambda表达式的主体
foreach (var n in multipliedNumbers)
{Console.WriteLine(n); //输出10 20 30 40 50
}//创建一个lambda表达式,并作为参数传递给OrderBy方法,按照数值大小排序
var sortedNumbers = numbers.OrderBy(n => n);//遍历LINQ查询结果,执行lambda表达式的主体foreach (var n in sortedNumbers)
{Console.WriteLine(n); //输出1 2 3 4 5
}
第三种:使用lambda表达式作为Task.Run或Task.Factory.StartNew的参数
Task是一种表示异步操作的对象,它可以用来实现并行编程、异步编程、多线程编程等功能。Task可以使用Task.Run或Task.Factory.StartNew方法来创建和启动,这些方法可以接受一个委托作为参数,表示要在任务中执行的代码。lambda表达式可以直接表示为委托的代码,而不需要单独定义一个方法。这样可以简化Task的创建和使用,提高代码的可读性和灵活性。
//创建一个lambda表达式,并作为参数传递给Task.Run方法,创建并启动一个任务
var task = Task.Run(() =>
{ //在任务中执行一些耗时的操作Thread.Sleep(1000);//返回一个结果return 42;
});//等待任务完成,获取任务的结果
int result = task.Result;
Console.WriteLine(result); //输出42//创建一个lambda表达式,并作为参数传递给Task.Factory.StartNew方法,创建并启动一个任务
var task2 = Task.Factory.StartNew(() =>
{//在任务中执行一些耗时的操作Thread.Sleep(1000);//抛出一个异常throw new Exception("Something went wrong");
});//等待任务完成,获取任务的异常
try
{task2.Wait();
}
catch (AggregateException ae)
{foreach (var e in ae.InnerExceptions){Console.WriteLine(e.Message); //输出Something went wrong}
}
第四种 使用lambda表达式作为表达式树
表达式树是一种表示代码结构的数据结构,它可以用来实现动态编译、元编程、LINQ提供程序等功能。表达式树可以使用Expression<TDelegate>类型来创建和表示,这是一种特殊的委托类型,它可以接受一个lambda表达式作为参数,表示要在表达式树中表示的代码。lambda表达式可以直接表示为表达式树的代码,而不需要单独定义一个方法。这样可以简化表达式树的创建和使用,提高代码的可读性和灵活性。
//创建一个lambda表达式,并作为参数传递给Expression<Func<int, int, int>>类型的构造函数,创建一个表达式树对象
Expression<Func<int, int, int>> expression = (x, y) => x + y;//遍历表达式树对象,获取表达式树的结构
Console.WriteLine(expression); //输出(x, y) => (x + y)
Console.WriteLine(expression.Body); //输出(x + y)
Console.WriteLine(expression.Parameters[0]); //输出x
Console.WriteLine(expression.Parameters[1]); //输出y//转换表达式树对象,编译为委托
Func<int, int, int> func = expression.Compile();//调用委托,执行lambda表达式的主体
Console.WriteLine(func(1, 2)); //输出3
第五种 使用lambda表达式作为异步方法
异步方法是一种使用async和await关键字来实现异步编程的方法,它可以用来实现非阻塞的UI、并发的网络请求、异步的IO操作等功能。异步方法可以使用Task或Task<T>类型来表示异步操作的结果,它们可以接受一个lambda表达式作为参数,表示要在异步方法中执行的代码。lambda表达式可以直接表示为异步方法的代码,而不需要单独定义一个方法。这样可以简化异步方法的创建和使用,提高代码的可读性和灵活性。
关键字来标记异步处理的代码,并使用await关键字来等待异步操作的完成。 - 将lambda表达式作为参数传递给Task或Task<T>类型的方法,如Task.Run或Task.FromResult,创建并启动一个异步方法。 - 等待异步方法完成,获取异步方法的结果或异常。例如,下面的代码演示了如何使用lambda表达式作为异步方法:
//创建一个lambda表达式,并作为参数传递给Task.Run方法,创建并启动一个异步方法
var task = Task.Run(async () =>
{//在lambda表达式中使用async和await关键字来标记异步处理的代码await Task.Delay(1000); //等待1秒return42; //返回一个结果
});//等待异步方法完成,获取异步方法的结果
int result = await task;
Console.WriteLine(result); //输出42//创建一个lambda表达式,并作为参数传递给Task.FromResult方法,创建并启动一个异步方法
var task2 = Task.FromResult(async () =>
{//在lambda表达式中使用async和await关键字来标记异步处理的代码await Task.Delay(1000); //等待1秒thrownew Exception("Something went wrong"); //抛出一个异常
});//等待异步方法完成,获取异步方法的异常try
{await task2;
}
catch (Exception e)
{Console.WriteLine(e.Message); //输出Something went wrong
}
二、匿名方法 简介在 C# 2.0 之前的版本中,我们创建委托的唯一形式 -- 命名方法。而 C# 2.0 -- 引进了匿名方法,在 ≥ C# 3.0 的版本中,我们会用 Lambda 表达式进行取代匿名方法,并且用 Lambda 表达式作为编写内联代码的首选方式,因为它更简洁。匿名方法是,顾名思义,匿名方法就是没有名称的方法。匿名方法最明显的好处就是:可以降低额外另写一个方法的工作量;另外一个好处就是可以直接访问调用者的变量,从而降低传参的复杂度。匿名方法,它不是一个事先定义的方法,而是使用一个委托的代码块,在使用时,和普通方法并没有什么区别,但是匿名方法可以在一定程度上减少系统开销。匿名方法,关键字:delegate。匿名方法,使用场景:通常在,①需要一个临时的方法,并且该方法使用的次数极少;②该方法的代码很短,不长。匿名方法的参数使用范围:匿名方法块。如果目标在块外部,你在匿名块内部使用 -- 跳转的关键字(类似 goto/break/continue),就是错误的;如果目标在块内部,你在匿名块外部使用 -- 跳转的关键字(类似 goto/break/continue),也是错误的。如果局部变量和参数的范围已经包含匿名方法声明,则该局部变量和参数将被称为该匿名方法的“外部”(外界)变量。你看,这个代码段中的 num 就是一个外部(外界)变量:int num = 250;MyDel del = delegate() { WriteLine("#:{0}", --num); };这个外部变量的所引用的 num,将会被认为是在创建委托时进行捕获的,它与我们常用的本地变量不同,这个外部变量的生存期 -- 引用该匿名方法的委托对象被 .NET 的 CLR 的垃圾回收机制进行回收。【注意】匿名方法无法访问外部(外界)范围内,带 ref 与 out 关键字的参数。【注意】在匿名块中不能访问 unsafe 的代码。【注意】在 is 运算符的左边,是用不了匿名的方法的。
三、匿名方法与Lambda表达式的区别
Lambda表达式和匿名方法的比较
第一、Lambda表达式本身就是匿名方法。
第二、Lambda表达式允许不指明参数类型,但是匿名方法必须要明确。
第三、Lambda表达式允许单一的表达式或多条语句组成,而匿名方法不允许单一表达式。
CalculatorDelegate cal2 = delegate (int a,int b)
{return a-b;
}public delegate int CalculatorDelegate(int a,int b);
CalculatorDelegate cal3 = (int a, int b) => { return a + b; };
CalculatorDelegate cal4 =(a,b)=>a-b;