委托、Lambda表达式、事件系列05,Action委托与闭包

来看使用Action委托的一个实例:

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

12

结果是期望能的2。但令人好奇的是:栈上的变量i是如何传递给Action委托的?

 

反编译进行查看,首先看Main方法对应的IL代码:

13

 

再看c_DisplayClass1的IL代码:

14

从中可以看出:
→在托管堆上创建了一个名为c_DisplayClass1的实例
→把栈上变量i的值赋值给了c_DisplayClass1的实例字段i
→编译器() => i++;Lambda表达式表示的匿名委托起了个<Main>b_0的方法名,并成为了c_DisplayClass1的实例方法
→把c_DisplayClass1的实例方法<Main>b_0赋值给Action委托变量
→最后调用委托2次,这2次都是针对c_DisplayClass1的实例字段i

 

换句话说,在托管堆上创建了对象实例,形成"闭包"。栈上的变量变成了闭包的实例字段,Lambda表达式所表示的匿名委托变成了闭包的实例方法。

 

以上,创建了一个Action,形成了一个"闭包",接下来创建2个Action,形成2个"闭包",看"闭包"的实例字段是否相互影响?

 
 
        static void Main(string[] args)
        {
            Action a = GetAction();
            Action b = GetAction();
 
 
            Console.Write("第一次调用a,i的值=");
            a();
            Console.WriteLine();
 
 
            Console.Write("第二次调用a,i的值=");
            a();
            Console.WriteLine();
 
 
            Console.Write("第一次调用b,i的值=");
            b();
            Console.WriteLine();
        }
 
 
        static Action GetAction()
        {
            Action result = null;
            int i = 0;
 
 
            result = () => Console.Write(i++);
            return result;          
        }
 
 
 
 

15

以上,虽然是把同一个GetAction方法分别赋值给了Action委托,但GetAction方法分别在不同的"闭包"内,当调用委托执行GetAction方法的时候,各自对闭包内的实例字段i自增1并打印,相互间不影响。

 

总结:每个Action都有自己的"闭包",并且"闭包"间互不影响。

 

 

“委托、Lambda表达式、事件系列”包括:

委托、Lambda表达式、事件系列01,委托是什么,委托的基本用法,委托的Method和Target属性

委托、Lambda表达式、事件系列02,什么时候该用委托

委托、Lambda表达式、事件系列03,从委托到Lamda表达式

委托、Lambda表达式、事件系列04,委托链是怎样形成的, 多播委托, 调用委托链方法,委托链异常处理

委托、Lambda表达式、事件系列05,Action委托与闭包

委托、Lambda表达式、事件系列06,使用Action实现观察者模式,体验委托和事件的区别

委托、Lambda表达式、事件系列07,使用EventHandler委托

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值