基础知识---委托和 lambda
委托定义类型,类型指定特定方法签名。 可将满足此签名的方法(静态或实例)分配给该类型的变量,然后(使用适当参数)直接调用该方法,或将其作为参数本身传递给另一方法再进行调用。 以下示例演示了委托的用法。
using System;
using System.Linq; public class Program
{
public delegate string Reverse(string s); static string ReverseString(string s)
{
return new string(s.Reverse().ToArray());
} static void Main(string[] args)
{
Reverse rev = ReverseString; Console.WriteLine(rev("a string"));
}
}
public delegate string Reverse(string s);
行创建特定签名的委托类型,在本例中即接受字符串参数并返回字符串参数的方法。static string ReverseString(string s)
方法与定义的委托类型具有完全相同的签名,用于实现委托。Reverse rev = ReverseString;
行显示可将方法分配给相应委托类型的变量。Console.WriteLine(rev("a string"));
行演示如何使用委托类型的变量来调用委托。
为简化开发过程,.NET 包含一组委托类型,程序员可重用这些类型而无需创建新类型。 其中包括 Func<>
、Action<>
和 Predicate<>
,可用于 .NET API 的各个位置,无需定义新委托类型。 当然,从这三者的签名可以看出它们之间存在某些差异,主要影响其既定用途:
Action<>
用于需要使用委托参数执行操作的情况。Func<>
通常用于现有转换的情况,也就是说需要将委托参数转换为其他结果时。 最好的示例就是投影。Predicate<>
用于需要确定参数是否满足委托条件的情况。 也可将其写作Func<T, bool>
。
现在可使用 Func<>
委托而非自定义类型重新编写上述示例。 程序将照旧继续运行。
using System;
using System.Linq; public class Program
{
static string ReverseString(string s)
{
return new string(s.Reverse().ToArray());
} static void Main(string[] args)
{
Func<string, string> rev = ReverseString; Console.WriteLine(rev("a string"));
}
}
对于此简单示例而言,在 Main
方法之外定义方法似乎有些多余。 因此 .NET Framework 2.0 引入了匿名委托的概念。 在其支持下,可创建“内联”委托,而无需指定任何其他类型或方法。 只需在所需位置内联委托的定义即可。
例如,要进行切换并使用匿名委托筛选出只有偶数的列表,然后将其打印到控制台。
using System;
using System.Collections.Generic; public class Program
{
public static void Main(string[] args)
{
List<int> list = new List<int>(); for (int i = ; i <= ; i++)
{
list.Add(i);
} List<int> result = list.FindAll(
delegate (int no)
{
return (no % == );
}
); foreach (var item in result)
{
Console.WriteLine(item);
}
}
}
如你所见,该委托的正文只是一组表达式,其他所有委托也是如此。 但它并非单独定义,而是在调用List<T>.FindAll 方法时临时引入。
但是,即使使用此方法,仍有许多可以丢弃的代码。 此时就需要使用 lambda 表达式。
lambda 表达式(或简称“lambda”)在 C# 3.0 中作为语言集成查询的 (LINQ) 核心构建基块被首次引入。 这种表达式只是使用委托的更方便的语法。 它们将声明签名和方法正文,但在分配到委托之前没有自己的正式标识。 与委托不同,可将其作为事件注册的左侧内容或在各种 LINQ 子句和方法中直接分配。
由于 lambda 表达式只是指定委托的另一种方式,因此应可重新编写上述示例,令其使用 lambda 表达式而不是匿名委托。
using System;
using System.Collections.Generic; public class Program
{
public static void Main(string[] args)
{
List<int> list = new List<int>(); for (int i = ; i <= ; i++)
{
list.Add(i);
} List<int> result = list.FindAll(i => i % == ); foreach (var item in result)
{
Console.WriteLine(item);
}
}
}
在前面的示例中,所使用的 Lambda 表达式为 i => i % 2 == 0
。 再次强调,它只是使用委托的一种非常方便的语法,因此其实际行为与使用匿名委托时相同。
再次强调,lambda 只是委托,这意味着可将其顺利用作事件处理程序,如以下代码片段所示。
public MainWindow()
{
InitializeComponent(); Loaded += (o, e) =>
{
this.Title = "Loaded";
};
}
此上下文中的 +=
运算符用于订阅事件。
最新文章
- 关于URLEnCode,URLDeCode,Base64,公钥私钥
- Tomcat中使用JNDI加载JDBC数据源
- ionic入门之AngularJS扩展(一)
- GCD常用剖析
- 28.USB的传输类型
- codeforces 691D Swaps in Permutation DFS
- hdu 1301
- 01 Mybatis 的配置和使用
- JAVA解析XML文件(DOM,SAX,JDOM,DOM4j附代码实现)
- hbase之认识
- servlet的url-pattern的缺省匹配【<;url-pattern>;/<;url-pattern>;】
- Spring的jdbc模板1
- [C++]Linux之头文件sys/types.h[/usr/include/sys]
- vs2010黑色主题Dark完美设置
- WPF参考
- Git常用命令及场景
- github上DQN代码的环境搭建,及运行(Human-Level Control through Deep Reinforcement Learning)conda配置
- iOS11 适配
- DVWA渗透测试系列 一 (DVWA环境配置)
- Redis List数据类型