C#Lambda表达式与委托关系

1. 核心关系图示

复制代码
A[委托] --> B[提供方法容器]
B --> C[Lambda表达式]
C --> D[委托实例的语法糖] 
A --> E[类型安全约束]
C --> F[编译器自动生成委托实例] 

2. 本质联系

2.1 类型关系

  • Lambda表达式 ‌是编译器生成的‌委托实例
  • 表达式自动匹配符合签名的委托类型
cs 复制代码
// 等效代码对比 
Func<int, int> square = x => x * x; 
// 编译器生成: 
Func<int, int> square = delegate(int x) { return x * x; }; 

2.2 编译机制

cs 复制代码
var list = new List<int> { 1, 2, 3 }; var evens = list.Where(n => n % 2 == 0); 
// 编译器处理为: 
IEnumerable<int> evens = Enumerable.Where(list, new Func<int, bool>(匿名方法对象)); 

3. 关键交互特性

3.1 类型推断

cs 复制代码
// 明确委托类型时 
Action<string> logger = msg => Console.WriteLine(msg); 
// 需显式声明类型的情况 
var processor = new SomeProcessor(); 
processor.DoWork((int x, int y) => x + y); 
// 无法推断参数类型时 

3.2 闭包现象

复制代码
void CreateActions(List<Action> actions) 
{ 
for (int i = 0; i < 3; i++) 
{ 
  actions.Add(() => Console.WriteLine(i)); 
} 
}
 // 输出全为3(闭包捕获变量引用)  

4. 应用场景对比

4.1 传统委托 vs Lambda

场景 传统委托写法 Lambda写法
按钮点击事件 button.Click += delegate { ... } button.Click += (s,e) => {...}
LINQ筛选 Where(delegate(int x) { return x>5; }) Where(x => x > 5)
异步回调 Task.Run(delegate { ... }) Task.Run(() => { ... })

4.2 特殊交互案例

cs 复制代码
// 多播委托中的Lambda 
Action multiAction = () => Console.Write("A"); 
multiAction += () => Console.Write("B");
multiAction(); 
// 输出AB(保留执行顺序) 
// 带返回值的Lambda 
Func<int> counter = () => {
 int count = 0; return ++count; 
}; 
Console.WriteLine(counter()); 
// 每次输出1(状态不保留) 

5. 底层原理分析

5.1 编译结果对比

cs 复制代码
// 源代码
 Func<int, bool> isEven = n => n % 2 == 0; 
// 反编译结果(部分) 
[CompilerGenerated] 
private sealed class <>c 
{ 
public static readonly <>c <>9 = new <>c();
 public static Func<int, bool> <>9__0_0; 
internal bool <Main>b__0_0(int n) 
{ return n % 2 == 0; } } 
// 实际调用 
Func<int, bool> isEven = <>c.<>9__0_0 ?? (<>c.<>9__0_0 = new Func<int, bool>(<>c.<>9.<Main>b__0_0)); 

5.2 内存模型

复制代码
 A[Lambda表达式] --> B[编译器生成密封类] 
B --> C[捕获的变量变为类的字段] 
B --> D[委托实例指向类方法] 

6. 最佳实践

6.1 选择依据

情况 推荐方式
简单单行逻辑 Lambda表达式
需要重用方法体 具名方法+委托
需要维护复杂状态 类实例方法

6.2 性能提示

cs 复制代码
// 避免高频调用的Lambda(每次生成新委托)
 for (int i = 0; i < 100000; i++) 
{ var temp = i; Task.Run(() => Process(temp)); 
// 产生大量临时委托 } 
// 优化方案(预先生成委托)
 static readonly Action<int> ProcessAction = Process;
 static void Process(int num) 
{ 
/*...*/ 
} 
// 调用 
Task.Run(() => ProcessAction(i)); 

7. 常见误区

7.1 延迟执行陷阱

var values = new[] { 1, 2, 3 };

var filters = new List<Func<int, bool>>();

for (int i = 0; i < 3; i++)

{ filters.Add(x => x > i);

// 捕获变化的i }

// 实际执行时i=3,所有条件变为x>3 Console.WriteLine(filters(2));

// 输出False

7.2 空值判断问题

EventHandler handler = null;

handler += (s, e) => Console.Write("A");

// 实际是handler = lambda handler?.Invoke(); // 可正常执行


总结图谱

A[Lambda表达式] --> B[委托实例]

B --> C[编译器生成类]

C --> D[捕获变量存储]

C --> E[方法指针绑定]

A --> F[类型系统]

F --> G[Action/Func]

F --> H[自定义委托]

相关推荐
她说彩礼65万7 小时前
C# params使用
开发语言·c#·log4j
专注VB编程开发20年7 小时前
C#内存加载dll和EXE是不是差不多,主要是EXE有入口点
数据库·windows·microsoft·c#
她说彩礼65万8 小时前
C# 反射
java·算法·c#
laocooon5238578869 小时前
C#二次开发中简单块的定义与应用
android·数据库·c#
YJlio9 小时前
BgInfo 学习笔记(11.5):多种输出方式(壁纸 / 剪贴板 / 文件)与“更新其他桌面”实战
笔记·学习·c#
Zhen (Evan) Wang9 小时前
.NET 6 API使用Serilog APM
c#·.net
wuk99810 小时前
C# 开发 FTP 客户端
开发语言·c#
咕白m62511 小时前
使用 C# 设置 Word 段落对齐样式
后端·c#
武藤一雄11 小时前
[.NET] 中 System.Collections.Generic命名空间详解
windows·微软·c#·asp.net·.net·.netcore
kingwebo'sZone11 小时前
一次找不到“无法加载dll 对应的”,多媒体没有启用(需要安装mediaplayer)
c#