c# 闭包的相关知识以及需要注意的地方

 更新时间:2020年06月20日 10:48:18   作者:老胡写代码  
这篇文章主要介绍了c# 闭包的相关知识以及需要注意的地方,文中讲解非常细致,代码帮助大家理解和学习,感兴趣的朋友可以参考下

虽然闭包主要是函数式编程的玩意儿,而C#的最主要特征是面向对象,但是利用委托或lambda表达式,C#也可以写出具有函数式编程风味的代码。同样的,使用委托或者lambda表达式,也可以在C#中使用闭包。

根据WIKI的定义,闭包又称语法闭包或函数闭包,是在函数式编程语言中实现语法绑定的一种技术。闭包在实现上是一个结构体,它存储了一个函数(通常是其入口地址)和一个关联的环境(相当于一个符号查找表)。闭包也可以延迟变量的生存周期。

嗯。。看定义好像有点迷糊,让我们看看下面的例子吧

  class Program
  {
    static Action CreateGreeting(string message)
    {
      return () => { Console.WriteLine("Hello " + message); };
    }

    static void Main()
    {
      Action action = CreateGreeting("DeathArthas");
      action();
    }
  }

这个例子非常简单,用lambda表达式创建一个Action对象,之后再调用这个Action对象。
但是仔细观察会发现,当Action对象被调用的时候,CreateGreeting方法已经返回了,作为它的实参的message应该已经被销毁了,那么为什么我们在调用Action对象的时候,还是能够得到正确的结果呢? 

原来奥秘就在于,这里形成了闭包。虽然CreateGreeting已经返回了,但是它的局部变量被返回的lambda表达式所捕获,延迟了其生命周期。怎么样,这样再回头看闭包定义,是不是更清楚了一些?

闭包就是这么简单,其实我们经常都在使用,只是有时候我们都不自知而已。比如大家肯定都写过类似下面的代码。

void AddControlClickLogger(Control control, string message)
{
	control.Click += delegate
	{
		Console.WriteLine("Control clicked: {0}", message);
	}
}

这里的代码其实就用了闭包,因为我们可以肯定,在control被点击的时候,这个message早就超过了它的声明周期。合理使用闭包,可以确保我们写出在空间和时间上面解耦的委托。

不过在使用闭包的时候,要注意一个陷阱。因为闭包会延迟局部变量的生命周期,在某些情况下程序产生的结果会和预想的不一样。让我们看看下面的例子。

  class Program
  {
	static List<Action> CreateActions()
    {
      var result = new List<Action>();
      for(int i = 0; i < 5; i++)
      {
        result.Add(() => Console.WriteLine(i));
      }
      return result;
    }

    static void Main()
    {
      var actions = CreateActions();
      for(int i = 0;i<actions.Count;i++)
      {
        actions[i]();
      }
    }
  }

这个例子也非常简单,创建一个Action链表并依次执行它们。看看结果

相信很多人看到这个结果的表情是这样的!!难道不应该是0,1,2,3,4吗?出了什么问题?

刨根问底,这儿的问题还是出现在闭包的本质上面,作为“闭包延迟了变量的生命周期”这个硬币的另外一面,是一个变量可能在不经意间被多个闭包所引用。

在这个例子里面,局部变量i同时被5个闭包引用,这5个闭包共享i,所以最后他们打印出来的值是一样的,都是i最后退出循环时候的值5。

要想解决这个问题也很简单,多声明一个局部变量,让各个闭包引用自己的局部变量就可以了。

	//其他都保持与之前一致
    static List<Action> CreateActions()
    {
      var result = new List<Action>();
      for (int i = 0; i < 5; i++)
      {
        int temp = i; //添加局部变量
        result.Add(() => Console.WriteLine(temp));
      }
      return result;
    }

这样各个闭包引用不同的局部变量,刚刚的问题就解决了。

除此之外,还有一个修复的方法,在创建闭包的时候,使用foreach而不是for。至少在C# 7.0 的版本上面,这个问题已经被注意到了,使用foreach的时候编译器会自动生成代码绕过这个闭包陷阱。

	//这样fix也是可以的
    static List<Action> CreateActions()
    {
      var result = new List<Action>();
      foreach (var i in Enumerable.Range(0,5))
      {
        result.Add(() => Console.WriteLine(i));
      }
      return result;
    }

这就是在闭包在C#中的使用和其使用中的一个小陷阱,希望大家能通过老胡的文章了解到这个知识点并且在开发中少走弯路!

以上就是c# 闭包的相关知识以及需要注意的地方的详细内容,更多关于c# 闭包的资料请关注脚本之家其它相关文章!

相关文章

  • c#创建Graphics对象的三种方法

    c#创建Graphics对象的三种方法

    通常我们使用下述三种方法来创建一个Graphics对象。
    2013-05-05
  • c# FTP上传文件实例代码(简易版)

    c# FTP上传文件实例代码(简易版)

    下面小编就为大家分享一篇c# FTP上传文件的实例代码,超简单哦~希望对大家有所帮助。一起跟随小编过来看看吧,
    2017-12-12
  • C#实现无限级联下拉列表框

    C#实现无限级联下拉列表框

    这篇文章主要为大家详细介绍了C#实现无限级联下拉列表框的相关资料,感兴趣的小伙伴们可以参考一下
    2016-03-03
  • C#中async和await的深入分析

    C#中async和await的深入分析

    Async/Await是C# 5引入的关键字,用以提高用户界面响应能力和对Web资源的访问能力,同时它使异步代码的编写变得更加容易,下面这篇文章主要给大家介绍了关于C#中async和await的相关资料,需要的朋友可以参考下
    2022-11-11
  • C#针对xml基本操作及保存配置文件应用实例

    C#针对xml基本操作及保存配置文件应用实例

    这篇文章主要介绍了C#针对xml基本操作及保存配置文件应用实例,包括了针对XML文件的定义、初始化、创建、以及增删改查等基础操作,并配有详细的实例加以说明,需要的朋友可以参考下
    2014-10-10
  • C#向数据库中插入或更新null空值与延迟加载lazy

    C#向数据库中插入或更新null空值与延迟加载lazy

    这篇文章介绍了C#向数据库中插入或更新null空值与延迟加载lazy,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-05-05
  • C# 脚本引擎RulesEngine的使用详解

    C# 脚本引擎RulesEngine的使用详解

    这篇文章主要介绍了C# 脚本引擎RulesEngine的使用方法,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下
    2021-02-02
  • C#引用类型作为方法的参数分析

    C#引用类型作为方法的参数分析

    这篇文章主要介绍了C#引用类型作为方法的参数分析,以实例的形式较为详细的分析了参数的传值问题,需要的朋友可以参考下
    2014-11-11
  • C#实现Word转换TXT的方法详解

    C#实现Word转换TXT的方法详解

    这篇文章主要为大家详细介绍了如何利用C#实现Word转换TXT的功能,文中的示例代码讲解详细,对我们学习C#有一定的帮助,感兴趣的小伙伴可以跟随小编一起了解一下
    2022-12-12
  • 调整C#中数组大小的方法

    调整C#中数组大小的方法

    数组存储多个相同类型的一种非常常用的数据结构,它长度是固定,也就是数组一旦创建大小就固定了,C# 数组不支持动态长度,那在C#中是否有方法可以调整数组大小呢?本文将通过示例介绍一种调整一维数组大小的方法,需要的朋友可以参考下
    2024-06-06

最新评论