C#闭包知识点详解

一闭包的定义

闭包(Closure) 是指一个函数(在 C# 中通常是 Lambda 或匿名方法)与其执行所需的非局部变量环境绑定的实体。

  • "外":外部方法提供的局部变量。

  • "内":内部的 Lambda 表达式。

  • "闭":当 Lambda 被传递到外部方法作用域之外时,它把那些局部变量"封闭"在了自己的包裹里,带着到处跑。

我们可以把这个特性总结为三句话,这三句话能帮你彻底看透闭包的"特权":

1. 访问权:打破"围墙"

正常情况下,函数之间是有"围墙"的。Main 里的变量 moodLevel 本该属于 Main 私有。 但一旦你写了闭包函数(那个 Lambda),它就像在围墙上打了个洞,直接伸手Main 的地盘拿东西用。

2. 携带权:自带"干粮"

这是最神奇的。普通的函数执行完,它所用的局部变量就销毁了。 但闭包函数被赋值给 calculator.CurrentStrategy 后,它就被带到了 PriceCalculator 类里。即便此时 Main 函数已经运行结束了,闭包函数依然随身带着 那个 moodLevel

逻辑: 我虽然离开了家(Main),但我把家里的存折(变量)装进兜里带走了。

3. 修改权:实时更新

闭包函数对外部变量不是简单的"拍照留念",而是**"实时连线"**。

  • 如果你在 Main 里改了 moodLevel = 100

  • 闭包函数下次执行时,拿到的就是 100

  • 反过来,如果闭包函数内部写了 moodLevel--Main 里的 moodLevel 也会跟着变。


⚠️ 必须注意的"代价"

虽然闭包能访问外部变量,但这也是有代价的,你一定要记住这一点:

内存占用(GC 无法回收): 因为闭包函数一直"拽着"外部变量不撒手,只要这个函数(委托对象)还在,那个外部变量就永远不会从内存里消失。

二触发情况

其实,闭包的触发条件非常简单,你可以记一个**"三要素"公式**。只要同时满足这三点,闭包就发生了:

闭包触发"三要素"

  1. 有嵌套:在一个方法(外部)里写了 Lambda 或匿名方法(内部)。

  2. 有引用 :内部方法用到了外部方法的局部变量(包括参数)。

  3. 有传递 :这个内部方法被返回 了,或者传给了其他方法,使得它比外部方法活得更久。

总结:

只要你看到**"逻辑被打包带走"**,就是闭包。

三底层原理

这是 0 基础理解闭包的关键。当你写了一个闭包,编译器会执行**"提升"(Hoisting)**操作:

  1. 生成隐藏类:编译器偷偷创建一个类(程序员看不见)。

  2. 搬家 :把被捕获的局部变量,从**栈(Stack)上搬到这个秘密类的字段(Field)**里。

  3. 实例化:在方法运行时,实例化这个类。由于类在**堆(Heap)**上,所以即便方法结束了,变量依然活着。

四必考知识点:捕获的是"变量"而非"值

这是闭包最著名的坑,我们通过代码来拆解:

错误示范(著名的 For 循环坑)

cs 复制代码
List<Action> actions = new List<Action>();

for (int i = 0; i < 3; i++)
{
    // 每一个 Action 都捕获了变量 i
    actions.Add(() => Console.WriteLine(i));
}

foreach (var a in actions) a();

结果: 打印出 3, 3, 3,而不是 0, 1, 2原因: 三个 Lambda 捕获的是同一个变量 i 的引用。当循环结束时,i 的值变成了 3,所以大家执行时看到的都是 3。

正确做法(创建副本)

cs 复制代码
for (int i = 0; i < 3; i++)
{
    int temp = i; // 每次循环都创建一个新的局部变量
    actions.Add(() => Console.WriteLine(temp));
}

结果: 0, 1, 2。因为每次循环的 temp 都是独立的变量,被不同的闭包实例分别捕获。

五闭包的优缺点分析

✅ 优点

  • 数据持久化:不需要定义全局变量,就能在多次调用间保持状态。

  • 简化逻辑:在回调函数中直接使用当前环境的数据,不需要通过参数传来传去。

  • 灵活架构:是实现"柯里化(Currying)"和"工厂模式"的基础。

❌ 缺点(风险)

  • 内存泄漏 :如果闭包捕获了一个巨大的对象(如 List),且委托一直不销毁,垃圾回收器(GC)就永远无法回收那个大对象。

  • 副作用:多个闭包修改同一个变量时,会产生难以调试的状态同步问题。

六什么时候用闭包

  • 动态配置逻辑:如练习 3 的问候器,根据初始参数生成不同的逻辑。

  • 异步编程 :在 TaskThread 中记住启动时的上下文。

  • LINQ 查询 :在 .Where(x => x > limit) 中,limit 往往是一个外部变量。

相关推荐
bug和崩溃我都要2 小时前
Qt 封装 libmpv 全功能视频播放器开发指南
开发语言·qt·音视频
郝学胜-神的一滴2 小时前
Qt 高级开发 018:复刻经典登录界面布局与窗口美化全解析
开发语言·c++·qt·程序人生·用户界面
郝亚军2 小时前
IEEE 754 单精度浮点的SEM表示
开发语言·c++·算法
zhangjw342 小时前
第15篇:Java多线程零基础入门,进程线程、线程创建方式、线程生命周期、线程安全彻底吃透
java·开发语言·面试
蝈理塘(/_\)大怨种2 小时前
类和对象 (上)
java·开发语言
csdn_aspnet2 小时前
EasyModbus 与 C# 集成
c#·modbus·easymodbus
小新1102 小时前
qt creator 将qInfo的输出日志写入日志文档,方便查看
开发语言·qt
hssfscv3 小时前
QT的学习记录1
开发语言·qt·学习
SunnyDays10113 小时前
Python操作Excel批注:从基础添加到高级自定义的完整指南
开发语言·python·excel
Yyyyyy~3 小时前
【C++】数组篇
开发语言·c++