原 c#中闭包的实现方法

闭包即闭包是指可以包含自由(未绑定到特定对象)变量的代码块.表现出来是调用函数结束后,函数内的变量的生存周期拉长到调用者的生命。很多闭包实现成匿名函数(js也是表现成匿名函数的,其他的方法不清楚),3.0中引入了匿名函数,相应的也提供了闭包的支持。

闭包实例:

    class Program
    {
        static void Main(string[] args)
        {
            Action ss = bibao();
            ss();
            Console.ReadKey();
        }
        public static Action bibao()
        {
            int i = 100;
            return delegate {
                Console.WriteLine(i);
            };
        }
    }

通常情况下{}内的变量在结束后会结束生命周期,然而在这里ss()却可以打印出i的值。生命周期从bibao函数里面拉长到Main里面。

在js里面是通过函数对象之间作用域链的引用关系实现,那么在c#中又是用什么方法实现的呢?

反编译代码:

编译后的代码生成了一个新的类,c#的闭包就是建立在这个类的基础上面的。

其中闭包中的变量作为类的公开成员变量,闭包函数自身作为成员,类型是internal。因为此类和闭包函数所在的类生成在一个同一个程序集中,而闭包流程中并不会使用这个类与其他程序集直接交流。

具体的调用过程

Main中:

.method private hidebysig static void  Main(string[] args) cil managed
{
  .entrypoint
  // Code size       17 (0x11)
  .maxstack  8
  IL_0000:  call       class [mscorlib]System.Action ExtendMwthod.Program::bibao()
  IL_0005:  callvirt   instance void [mscorlib]System.Action::Invoke()
  IL_000a:  call       valuetype [mscorlib]System.ConsoleKeyInfo [mscorlib]System.Console::ReadKey()
  IL_000f:  pop
  IL_0010:  ret
} // end of method Program::Main

直接调用的依然是bibao方法,再invoke,没发现原因。

.method public hidebysig static class [mscorlib]System.Action 
        bibao() cil managed
{
  // Code size       25 (0x19)
  .maxstack  8
  IL_0000:  newobj     instance void ExtendMwthod.Program/'<>c__DisplayClass1_0'::.ctor()
  IL_0005:  dup
  IL_0006:  ldc.i4.s   100
  IL_0008:  stfld      int32 ExtendMwthod.Program/'<>c__DisplayClass1_0'::i
  IL_000d:  ldftn      instance void ExtendMwthod.Program/'<>c__DisplayClass1_0'::'<bibao>b__0'()
  IL_0013:  newobj     instance void [mscorlib]System.Action::.ctor(object,
                                                                    native int)
  IL_0018:  ret
} // end of method Program::bibao

可以看到自动生成的类在这里实例化了,闭包函数内部已经改写了。起始用字段i和方法'<bibao>b__0'实例化了action,因而在main中调用的时候变量已经包含在action的参数里面带过去了。通过这种方法实现了变量生命周期的延长。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏mwangblog

python函数

1332
来自专栏逸鹏说道

Python3 与 C# 基础语法对比(Function专栏-新排版)

在线编程:https://mybinder.org/v2/gh/lotapp/BaseCode/master

1353
来自专栏Script Boy (CN-SIMO)

几个整数求和

任务描述 在eclipse中通过配置运行参数,编写程序求这些整数参数之和然后输出参数个数与所求之和。 设计思想 获取的参数是String类型的,通过Intege...

2260
来自专栏LinkedBear的个人空间

唠唠SE的集合-00——概述 原

                        由于是数组实现,在增和删的时候会牵扯到数组增容、以及拷贝元素,所以慢。

872
来自专栏Golang语言社区

GO-回调函数

1,什么是回调函数。一个函数中有个参数为函数的指针。该函数自己调用自己指针函数即可; 2,为什么要使用回调函数。抽象画了一个函数,简洁代码。异步执行 3,例子:...

3315
来自专栏小文博客

写出这个数——《C语言代码笔记》

2212
来自专栏Java学习网

Java中正则表达式分类及使用方法,分门别类,总结的很好

1、正则表达式是什么? 是指一个用来描述或者匹配一系列符合某个语法规则的字符串的单个字符串。其实就是一种规则。有自己特殊的应用。 作用 : 比如注册邮箱 , ...

2836
来自专栏恰同学骚年

《C#图解教程》读书笔记之五:委托和事件

  (1)本质:持有一个或多个方法的对象;委托和典型的对象不同,执行委托实际上是执行它所“持有”的方法。如果从C++的角度来理解委托,可以将其理解为一个类型安全...

802
来自专栏闪电gogogo的专栏

Python——正则表达式

此篇文章结合小甲鱼的笔记和视频整理。 1 编译 Python 通过 re 模块为正则表达式引擎提供一个接口,同时允许你将正则表达式编译成模式对象,并用它们来进行...

27210
来自专栏yl 成长笔记

c# typeof 与 GetType 作用与区别

Used to obtain the "System.Type" object for a type. A 'typeof‘ expression takes ...

2451

扫码关注云+社区