C#闭包
程序员文章站
2022-07-11 10:54:47
1.匿名方法和Lambda表达式中可以访问到声明该匿名方法或Lambda表达式所在方法中的参数或局部变量,这些变量称为外部变量(Outer Variable),外部变量的生命周期会持续到引用匿名方法或Lambda表达式的委托实例被垃圾回收器回收为止: 2.调用匿名方法或Lambda表达式时,访问到的 ......
1.匿名方法和lambda表达式中可以访问到声明该匿名方法或lambda表达式所在方法中的参数或局部变量,这些变量称为外部变量(outer variable),外部变量的生命周期会持续到引用匿名方法或lambda表达式的委托实例被垃圾回收器回收为止:
int mynum = 1; action myaction = () => { console.writeline(mynum); //此处的mynum即该lambda表达式的一个外部变量 };
2.调用匿名方法或lambda表达式时,访问到的是外部变量最终的值,而不是声明该匿名方法或lambda表达式时的值;同样,在匿名方法或lambda表达式中修改外部变量的值后,在外部访问到的也是修改后的值;这个特性被称为闭包(closure);
例如:
action myaction = null; for (int i = 0; i < 5; i++) { myaction += () => { console.write(i); //由于lambda表达式调用了该变量i,此处会将i提升为一个字段,该字段最后取值为5 //或使用以下方式: //int x = i; //console.write(x); }; } myaction(); //55555 myaction = null; for (int i = 0; i < 5; i++) { int closureindex = i; //每次都声明一个新的局部变量,由于lambda表达式内调用了该新声明的变量closureindex,此处相当于每次都会声明一个新的字段,并且互不影响取值 myaction += () => { console.write(closureindex); }; } myaction(); //01234 myaction = null; for (int i = 0; i < 5; i++) { int closureindex = i; myaction +=() => { console.write(closureindex); }; ++closureindex; } myaction(); //12345
2.实际上,在声明匿名方法和lambda表达式时,如果在方法或表达式内部调用了外部变量,编译器会创建一个以<>c__displayclass开头的匿名嵌套类,它包含该方法或表达式内调用的所有外部变量的字段,运行时会创建该匿名嵌套类的一个实例,并将外部变量的值传入该实例对应的字段,外部对该变量的所有操作也会变成对该嵌套类实例中字段的操作;
此处以外部变量为值类型举例,对于引用类型,匿名嵌套类中会保存该引用类型的对象:
class program { static void main(string[] args) { int mynum = 1; action myaction = () => { console.writeline(mynum); }; mynum = 2; myaction(); console.read(); } }
如果您觉得阅读本文对您有帮助,请点一下“推荐”按钮,您的认可是我写作的最大动力!
作者:minotauros
出处:
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。