匿名方法定义
匿名方法的声明
匿名方法的使用
范围捕获局部变量或实例状态
匿名函数会捕获变量,并延长对象的生命周期
结束匿名方法捕获局部变量-静态匿名函数
匿名方法是没有名称只有主体的方法。 在匿名方法中您不需要指定返回类型,它是从方法主体内的 return 语句推断的。 匿名方法是通过使用 delegate 关键字创建委托实例来声明的。
第一种方式:<code></code>
第二种方式:可以将创建的匿名方法转换为具有任何参数列表的委托类型。这是 lambda 表达式不支持的匿名方法的唯一功能。 在所有其他情况下,lambda 表达式是编写内联代码的首选方法。
从 C# 9.0 开始,可以使用弃元指定该方法不使用的两个或更多个匿名方法输入参数:
<code></code>为实现向后兼容性,如果只有一个参数名为 <code>_</code>,则将 <code>_</code> 视为匿名方法中该参数的名称。
从 C# 9.0 开始,可以在匿名方法的声明中使用 <code>static</code> 修饰符:
静态匿名方法无法从封闭范围捕获局部变量或实例状态。
总结:
匿名函数会捕获当前上下文的局部变量,捕获是引用地址不是赋值,延长对象的生命周期; 捕获的变量将不会被作为垃圾回收,直到此委托或表达式树被回收掉。
这边说的当前上下文就是private public interal protected声明的那些指函数、变量等多作用域范围
也就是说,只要某个方法中存在没有被回收的匿名函数/lambda 表达式/表达式树,那么当前上下文的对象直到这些匿名函数被回收之前都不会被回收,即便已经设为了 null。
案例:
测试代码是这样的:
需要验证的是 <code>MainPage</code> 对象是否被回收。然而在这段代码中,<code>MainPage</code> 并没有被回收;然而去掉最后一行,<code>MainPage</code> 便可以正常回收。关键是,即便是在 Console.WriteLine 上打下断点,让代码永远不会执行到最后一句,也不会改变回收的结果。
由于 <code>DoSomething</code> 中的委托参数恰好就是 <code>MainPage</code> 类型的,不禁让人觉得可能是此函数做了一些奇怪的事情。然而毕竟参数中传入的委托参数只是形参,理论上不应该影响到外部对象的回收。那么影响的只可能是变量的捕获了。
于是,我们将最后一行换成别的函数别的参数:
或者将整个这一句提取成新的函数:
那么,回收就会正常进行。
现在,不执行这个受争议的函数了,我们使用空的匿名函数。
一样会导致不回收。
在微软官方的《C# 规范 5.0》(点此下载)的第 7.15.5.1 章节中有说到:
When an outer variable is referenced by an anonymous function, the outer variable is said to have been captured by the anonymous function. Ordinarily, the lifetime of a local variable is limited to execution of the block or statement with which it is associated (§5.1.7). However, the lifetime of a captured outer variable is extended at least until the delegate or expression tree created from the anonymous function becomes eligible for garbage collection.

匿名函数会捕获当前上下文的局部变量,延长对象的生命周期;直到此委托或表达式树被回收掉。
您可以在lambda或匿名方法上利用static修饰符,以确保您不会无意中从封闭的上下文中捕获本地或实例状态。这对于提高应用程序的性能非常友好。
编程是个人爱好