C#委托内部使用局部的变量的问题
一. 引子
先来看如下代码:
int i = ;
Action action1 = () =>
{
Console.WriteLine("打印一下i的值:" + i);
};
i = ;
Action action2 = () =>
{
Console.WriteLine("打印一下i的值:" + i);
};
action1.Invoke();
action2.Invoke();
那么问题来了,执行这两个委托之后,输出的是
打印一下i的值:0
打印一下i的值:1
还是
打印一下i的值:1
打印一下i的值:1
相信有一些人会觉的是第一种情况,但是实际的结果却是第二种情况的结果,如下图测试结果,这是为什么呢?

二. 原理部分
首先我们会议一下C#中委托究竟是什么?
相信很多的小伙伴都知道,委托是一种语法糖。它在编辑之后,编译器会为其生成一个类。我们使用 ILSpy.exe 来看一下编译之后生成的是什么:
ILSpy.exe 打开,如下图:

这个自动生成的类 "<>c__DisplayClass0_0" 便是委托生成的类,看一下IL代码:
.class nested private auto ansi sealed beforefieldinit '<>c__DisplayClass0_0'
extends [mscorlib]System.Object
{
.custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = (
)
// Fields
.field public int32 i // Methods
.method public hidebysig specialname rtspecialname
instance void .ctor () cil managed
{
// Method begins at RVA 0x20a0
// Code size 8 (0x8)
.maxstack IL_0000: ldarg.
IL_0001: call instance void [mscorlib]System.Object::.ctor()
IL_0006: nop
IL_0007: ret
} // end of method '<>c__DisplayClass0_0'::.ctor .method assembly hidebysig
instance void '<Main>b__0' () cil managed
{
// Method begins at RVA 0x20a9
// Code size 29 (0x1d)
.maxstack // (no C# code)
IL_0000: nop
// Console.WriteLine("打印一下i的值:" + this.i);
IL_0001: ldstr "打印一下i的值:"
IL_0006: ldarg.
IL_0007: ldfld int32 Demo.Program/'<>c__DisplayClass0_0'::i
IL_000c: box [mscorlib]System.Int32
IL_0011: call string [mscorlib]System.String::Concat(object, object)
IL_0016: call void [mscorlib]System.Console::WriteLine(string)
// (no C# code)
IL_001b: nop
IL_001c: ret
} // end of method '<>c__DisplayClass0_0'::'<Main>b__0' .method assembly hidebysig
instance void '<Main>b__1' () cil managed
{
// Method begins at RVA 0x20a9
// Code size 29 (0x1d)
.maxstack // (no C# code)
IL_0000: nop
// Console.WriteLine("打印一下i的值:" + this.i);
IL_0001: ldstr "打印一下i的值:"
IL_0006: ldarg.
IL_0007: ldfld int32 Demo.Program/'<>c__DisplayClass0_0'::i
IL_000c: box [mscorlib]System.Int32
IL_0011: call string [mscorlib]System.String::Concat(object, object)
IL_0016: call void [mscorlib]System.Console::WriteLine(string)
// (no C# code)
IL_001b: nop
IL_001c: ret
} // end of method '<>c__DisplayClass0_0'::'<Main>b__1' } // end of class <>c__DisplayClass0_0
我们可以看到,这个自动生成的类中,它有两个方法 "<Main>b__0" 和 "<Main>b__1" ,分别对应的是委托 action1 和 action2,于此同时还有一个 public 的成员字段 i,用来保存外部传递的 i 的值。很明显,由于第二次的赋值,把 i 赋值成了1,所以会导致两个方法的 i 的值因为共用一个成员变量而都成为1。
三. 解决方案
那么问题来了,我们怎么样才能用局部的临时变量同时赋值给两个委托而对它们不会造成互相影响呢?
1. 设置连个不同的临时变量(感觉有点笨)
程序代码:
int i = ;
Action action1 = () =>
{
Console.WriteLine("打印一下i的值:" + i);
};
int j = ;
Action action2 = () =>
{
Console.WriteLine("打印一下i的值:" + j);
};
action1.Invoke();
action2.Invoke();
2. 将临时变量的值按照委托的参数传入
程序代码:
Action<int> action1 = (num) =>
{
Console.WriteLine("打印一下i的值:" + num);
};
action1.Invoke();
action1.Invoke();
C#委托内部使用局部的变量的问题的更多相关文章
- C# 中的局部static变量
其实这问题没什么可讨论的,C#不支持局部静态变量. 但还是想了一下C#为什么不支持局部静态变量,以下均是个人想法. C++和C支持局部静态变量,也就是在一个函数的内部声明一个静态变量,这种变量的特定如 ...
- c# 委托内部构造
以下纯属个人简介,错误之处,请随意指出. 委托是指向方法的,而事件是委托的触发器,执行事件,就会遍历委托里的方法,并且执行. 委托内部构造第一块是方法指针(methodPtr),用于指向方法的内存地址 ...
- C++中的局部变量、全局变量、局部静态变量、全局静态变量的区别
局部变量(Local variables)与 全局变量: 在子程序或代码块中定义的变量称为局部变量,在程序的一开始定义的变量称为全局变量. 全局变量作用域是整个程序,局部变量作用域是定义该变量的子程序 ...
- (八)Activiti之流程变量和局部流程变量
一.流程变量 1.1 概念 如果,当流程走到"学生请假"这个任务节点的时候,此时可以用TaskService设置流程变量,变量值包含请假人.请假时间.请假理由等信息,这些信息存在表 ...
- test dword ptr [eax],eax ; probe page.局部数组变量定义所分配的最大空间为1M
问题的出现 使用VS2017编写程序时,程序编译可以通过,但运行时就会弹出错误 经过查证发现: 这跟局部数组变量定义所分配的最大空间设置大小有关. 局部变量的申请空间是存放于栈中,windows里默认 ...
- c++ 反汇编 局部静态变量
vs2017下测试 34: for (int i = 0; i < 5; i++) 0029734E C7 45 F8 00 00 00 00 mov dword ptr [ebp-8],0 0 ...
- Java基础-内部类-为什么局部和匿名内部类只能访问局部final变量
先看下面这段代码: public class Test { public static void main(String[] args) { } public void test(final int ...
- [ python ] 全局和局部作用域变量的引用
全局与局部变量的引用 (a)locals(b)globals 这里还需要在补充2个关键字一起比较学习,关键字:(c)nonlocal(d)global locals 和 globals locals: ...
- C++函数返回局部指针变量
遇到过好几次关于函数返回指针变量问题,有时候是可以的,有时候是不可以的,然后就混乱了.今天研究了下,结果发现原来和内存分配有关. 用下面的例子分析下吧: char * test() { char a[ ...
随机推荐
- Python3字典update()方法
描述 Python字典update()函数把字典参数dict2的key/value(键/值)对更新到字典dict里. update()方法语法: dict.update(dict2) 参数 dict2 ...
- Web前端基础(17):jQuery基础(四)
1. jQuery的属性操作 jquery的属性操作模块分为四个部分:html属性操作,dom属性操作,类样式操作和值操作 html属性操作:是对html文档中的属性进行读取,设置和移除操作.比如at ...
- MySQL触发器学习总结
1.What 触发器是MySQL响应DELETE,INSERT,UPDATE语句前后而自动执行的一条MySQL语句 2.Why(使用情形) 增加一个订单对应库存-1 删除一行在 ...
- FCC---Use the CSS Transform scale Property to Scale an Element on Hover
The transform property has a variety of functions that let you scale, move, rotate, skew, etc., your ...
- scrapy实例:爬取天气、气温等
1.创建项目 scrapy startproject weather # weather是项目名称 scrapy crawl spidername开始运行,程序自动使用start_urls构造Requ ...
- Cobalt Strike系列教程第二章:Beacon详解
上周更新了Cobalt Strike系列教程第一章:简介与安装,文章发布后,深受大家的喜爱,遂将该系列教程的其他章节与大家分享,提升更多实用技能! 第二章:Beacon详解 一.Beacon命令 大家 ...
- [Go] 在golang中使用正则表达式捕获子表达式
正则匹配并且可以捕获到()这个里面的子表达式的值,linux的grep命令没办法捕获子表达式的值,只能获取到整条正则匹配的内容 package main import "regexp&quo ...
- NXP_RTCESL库
恩智浦实时控制嵌入式软件库(缩写为RTCESL,以前为恩智浦嵌入式软件库FSLESL)是一组算法,从基础数学运算到高级数学变换以及高级观测器,这些都可以方便地用在复杂的实时控制应用中以及我们的电机控制 ...
- Gaussian field consensus论文解读及MATLAB实现
Gaussian field consensus论文解读及MATLAB实现 作者:凯鲁嘎吉 - 博客园 http://www.cnblogs.com/kailugaji/ 一.Introduction ...
- JUC-4-CopyOnWriteArrayList
什么是CopyOnWrite容器 CopyOnWrite容器即写时复制的容器.通俗的理解是当我们往一个容器添加元素的时候,不直接往当前容器添加,而是先将当前容器进行Copy,复制出一个新的容器,然后新 ...