欢迎访问 生活随笔!

凯发k8官方网

当前位置: 凯发k8官方网 > 编程语言 > c# >内容正文

c#

c# 睡眠3秒-凯发k8官方网

发布时间:2024/10/8 c# 0 豆豆
凯发k8官方网 收集整理的这篇文章主要介绍了 c# 睡眠3秒_c#中的闭包和意想不到的坑 小编觉得挺不错的,现在分享给大家,帮大家做个参考.
转自:老胡写代码cnblogs.com/deatharthas/p/13166987.html

虽然闭包主要是函数式编程的玩意儿,而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 listcreateactions()
{
var result = new list();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[i]();
}
}
}

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

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

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

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

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

//其他都保持与之前一致
static listcreateactions()
{var result = new list();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 createactions()
{var result = new list();foreach (var i in enumerable.range(0,5))
{
result.add(() => console.writeline(i));
}return result;
}

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

往期精彩回顾

【推荐】.net core开发实战视频课程 ★★★

.net core实战项目之cms 第一章 入门篇-开篇及总体规划

【.net core微服务实战-统一身份认证】开篇及目录索引

redis基本使用及百亿数据量中的使用技巧分享(附视频地址及观看指南)

.net core中的一个接口多种实现的依赖注入与动态选择看这篇就够了

10个小技巧助您写出高性能的asp.net core代码

用abp vnext快速开发quartz.net定时任务管理界面

在asp.net core中创建基于quartz.net托管服务轻松实现作业调度

现身说法:实际业务出发分析百亿数据量下的多表查询优化

关于c#异步编程你应该了解的几点建议

c#异步编程看这篇就够了

给我好看

您看此文用

  · 

秒,转发只需1秒呦~

好看你就

点点

总结

以上是凯发k8官方网为你收集整理的c# 睡眠3秒_c#中的闭包和意想不到的坑的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得凯发k8官方网网站内容还不错,欢迎将凯发k8官方网推荐给好友。

  • 上一篇:
  • 下一篇:
网站地图