C#中闭包概念讲解

作者:黑洞视界 时间:2022-08-16 05:16:28 

理解C#中的闭包

1、 闭包的含义

首先闭包并不是针对某一特定语言的概念,而是一个通用的概念。除了在各个支持函数式编程的语言中,我们会接触到它。一些不支持函数式编程的语言中也能支持闭包(如java8之前的匿名内部类)。

在看过的对于闭包的定义中,个人觉得比较清晰的是在《JavaScript高级程序设计》这本书中看到的。具体定义如下:

闭包是指有权访问另一个函数作用域中的变量的函数。

注意,闭包这个词本身指的是一种函数。而创建这种特殊函数的一种常见方式是在一个函数中创建另一个函数。

2、 在C# 中使用闭包(例子选取自《C#函数式程序设计》)

下面我们通过一个简单的例子来理解C#闭包


class Program
{
   static void Main(string[] args)
   {
       Console.WriteLine(GetClosureFunction()(30));
   }

static Func<int, int> GetClosureFunction()
   {
       int val = 10;
       Func<int, int> internalAdd = x => x + val;

Console.WriteLine(internalAdd(10));

val = 30;
       Console.WriteLine(internalAdd(10));

return internalAdd;
   }
}

上述代码的执行流程是Main函数调用GetClosureFunction函数,GetClosureFunction返回了委托internalAdd并被立即执行了。

输出结果依次为20、40、60

对应到一开始提出的闭包的概念。这个委托internalAdd就是一个闭包,引用了外部函数GetClosureFunction作用域中的变量val。

注意:internalAdd有没有被当做返回值和闭包的定义无关。就算它没有被返回到外部,它依旧是个闭包。

3、 理解闭包的实现原理

我们来分析一下这段代码的执行过程。在一开始,函数GetClosureFunction内定义了一个局部变量val和一个利用lamdba语法糖创建的委托internalAdd。

第一次执行委托internalAdd 10 + 10 输出20

接着改变了被internalAdd引用的局部变量值val,再次以相同的参数执行委托,输出40。显然局部变量的改变影响到了委托的执行结果。

GetClosureFunction将internalAdd返回至外部,以30作为参数,去执行得到的结果是60,和val局部变量最后的值30是一致的。

val 作为一个局部变量。它的生命周期本应该在GetClosureFunction执行完毕后就结束了。为什么还会对之后的结果产生影响呢?

我们可以通过反编译来看下编译器为我们做的事情。

为了增加可读性,下面的代码对编译器生成的名字进行修改,并对代码进行了适当的整理。


class Program
{
   sealed class DisplayClass
   {
       public int val;

public int AnonymousFunction(int x)
       {
           return x + this.val;
       }
   }

static void Main(string[] args)
   {
       Console.WriteLine(GetClosureFunction()(30));
   }

static Func<int, int> GetClosureFunction()
   {
       DisplayClass displayClass = new DisplayClass();
       displayClass.val = 10;
       Func<int, int> internalAdd = displayClass.AnonymousFunction;

Console.WriteLine(internalAdd(10));

displayClass.val = 30;
       Console.WriteLine(internalAdd(10));

return internalAdd;
   }
}

编译器创建了一个匿名类(如果不需要创建闭包,匿名函数只会是与GetClosureFunction生存在同一个类中,并且委托实例会被缓存,参见clr via C# 第四版362页),并在GetClosureFunction中创建了它实例。局部变量实际上是作为匿名类中的字段存在的。

4、 C#7对于不作为返回值的闭包的优化

如果在vs2017中编写第二节的代码。会得到一个提示,询问是否把lambda表达式(匿名函数)托转为本地函数。本地函数是c#7提供的一个新语法。那么使用本地函数实现闭包又会有什么区别呢?

如果还是第二节那样的代码,改成本地函数,查看IL代码。实际上不会发生任何变化。


class Program
{
   static void Main(string[] args)
   {
       Console.WriteLine(GetClosureFunction()(30));
   }

static Func<int, int> GetClosureFunction()
   {
       int val = 10;
       int InternalAdd(int x) => x + val;

Console.WriteLine(InternalAdd(10));

val = 30;
       Console.WriteLine(InternalAdd(10));

return InternalAdd;
   }
}

但是当internalAdd不需要被返回时,结果就不一样了。

下面分别来看下匿名函数和本地函数创建不作为返回值的闭包的时候演示代码及经整理的反编译代码。

匿名函数


static void GetClosureFunction()
{
   int val = 10;
   Func<int, int> internalAdd = x => x + val;

Console.WriteLine(internalAdd(10));

val = 30;
   Console.WriteLine(internalAdd(10));
}

经整理的反编译代码


sealed class DisplayClass
{
   public int val;

public int AnonymousFunction(int x)
   {
       return x + this.val;
   }
}

static void GetClosureFunction()
{
   DisplayClass displayClass = new DisplayClass();
   displayClass.val = 10;
   Func<int, int> internalAdd = displayClass.AnonymousFunction;

Console.WriteLine(internalAdd(10));

displayClass.val = 30;
   Console.WriteLine(internalAdd(10));
}

本地函数


class Program
{
   static void Main(string[] args)
   {
   }

static void GetClosureFunction()
   {
       int val = 10;
       int InternalAdd(int x) => x + val;

Console.WriteLine(InternalAdd(10));

val = 30;
       Console.WriteLine(InternalAdd(10));
   }
}

经整理的反编译代码


// 变化点1:由原来的class改为了struct
struct DisplayClass
{
   public int val;

public int AnonymousFunction(int x)
   {
       return x + this.val;
   }
}

static void GetClosureFunction()
{
   DisplayClass displayClass = new DisplayClass();
   displayClass.val = 10;

// 变化点2:不再构建委托实例,直接调用值类型的实例方法
   Console.WriteLine(displayClass.AnonymousFunction(10));

displayClass.val = 30;
   Console.WriteLine(displayClass.AnonymousFunction(10));
}

上述这两点变化在一定程度上能够带来性能的提升,目前的理解是,用结构体代替类,结构体实例能够在方法跑完后就立即释放,不需要等待垃圾回收,所以在官方的推荐中,如果委托的使用不是必要的,更推荐使用本地函数而非匿名函数。

来源:https://www.cnblogs.com/blurhkh/p/9535289.html

标签:C#,闭包,函数
0
投稿

猜你喜欢

  • 一文带你了解Java万物之基之Object类

    2023-10-09 01:58:28
  • SpringBoot项目在IntelliJ IDEA中如何实现热部署

    2023-10-29 13:30:22
  • Spring Boot自定义Starter组件开发实现配置过程

    2022-05-31 16:07:12
  • Java调用Oracle存储过程详解

    2022-09-09 03:19:45
  • Hadoop组件简介

    2023-08-20 14:07:00
  • Java读取json数据并存入数据库的操作代码

    2023-09-23 06:00:57
  • MyBatis-Plus中更新操作的两种实现

    2022-06-24 03:09:14
  • Spring Boot @Conditional注解用法示例介绍

    2023-04-18 22:51:51
  • java 实现MD5加密算法的简单实例

    2023-07-19 21:53:56
  • 深入解析Java的Spring框架中bean的依赖注入

    2023-12-20 18:50:52
  • Java 反射调用静态方法的简单实例

    2021-10-09 06:47:58
  • 关于Java反编译字节码文件

    2021-10-07 01:41:32
  • 设计模式在Spring框架中的应用汇总

    2023-10-22 19:20:09
  • 教你快速搭建sona服务及idea使用sona的方法

    2023-11-20 05:22:53
  • 深入理解java final不可变性

    2023-02-11 20:17:27
  • IDEA 2019.2.3破解激活教程(亲测有效)

    2023-02-21 22:16:23
  • springboot vue组件开发实现接口断言功能

    2023-11-12 10:26:53
  • Spring Boot Logback配置日志过程解析

    2022-12-09 18:08:06
  • MyBatis JdbcType 与Oracle、MySql数据类型对应关系说明

    2023-08-23 02:23:06
  • springboot如何使用logback-spring配置日志格式,并分环境配置

    2023-11-10 04:37:34
  • asp之家 软件编程 m.aspxhome.com