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 往往是一个外部变量。

相关推荐
东北甜妹2 小时前
Redis Cluster 操作命令
java·开发语言
花间相见2 小时前
【大模型微调与部署01】—— ms-swift-3.12入门:安装、快速上手
开发语言·ios·swift
techdashen2 小时前
Rust 正式成立 Types Team:类型系统终于有了专属团队
开发语言·后端·rust
jiayong232 小时前
第 17 课:任务选择与批量操作
开发语言·前端·javascript·vue.js·学习
量子炒饭大师2 小时前
【C++11】RAII 义体加装指南 ——【包装器 与 异常】C++11中什么是包装器?有哪些包装器?C++常见异常有哪些?(附带完整代码讲解)
开发语言·c++·c++11·异常·包装器
telllong2 小时前
Python异步编程从入门到不懵:asyncio实战踩坑指南
开发语言·python
知兀2 小时前
【Result类】(使用/不使用<T> data的情况);自带静态方法、纯数据类;
java·开发语言
达帮主2 小时前
25.C语言 递归函数
c语言·开发语言·汇编
整点薯条7782 小时前
用 Python 给家里做一次噪音频谱审计:程序员的声学工程实践(含完整源码)
开发语言·python·噪音控制