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[自定义委托]

相关推荐
闪电麦坤953 小时前
C#:Time.deltaTime
开发语言·c#
InCerry9 小时前
.NET周刊【3月第3期 2025-03-16】
c#·asp.net·.net
观无11 小时前
C# 扩展方法
开发语言·c#
画个逗号给明天"13 小时前
C#从入门到精通(4)
数据库·c#
观无16 小时前
c#中的virtual方法
开发语言·c#
code bean17 小时前
【C#】ForEach vs foreach
开发语言·c#
OpenSeek18 小时前
【设计模式】面向对象的设计模式概述
设计模式·c#·设计原则
码观天工19 小时前
10年+ .NET Coder 心语 ── 继承的思维:从思维模式到架构设计的深度解析
c#·.net·继承·思维·面相对象
FAREWELL000751 天前
C#核心学习(一)面向过程与面向对象编程---初识类和对象
学习·c#·面向对象
yngsqq1 天前
Visual Studio中创建和配置设置文件(Settings.settings) - 详细步骤指南——待调试
c#