一. 引子

先来看如下代码:

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#委托内部使用局部的变量的问题的更多相关文章

  1. C# 中的局部static变量

    其实这问题没什么可讨论的,C#不支持局部静态变量. 但还是想了一下C#为什么不支持局部静态变量,以下均是个人想法. C++和C支持局部静态变量,也就是在一个函数的内部声明一个静态变量,这种变量的特定如 ...

  2. c# 委托内部构造

    以下纯属个人简介,错误之处,请随意指出. 委托是指向方法的,而事件是委托的触发器,执行事件,就会遍历委托里的方法,并且执行. 委托内部构造第一块是方法指针(methodPtr),用于指向方法的内存地址 ...

  3. C++中的局部变量、全局变量、局部静态变量、全局静态变量的区别

    局部变量(Local variables)与 全局变量: 在子程序或代码块中定义的变量称为局部变量,在程序的一开始定义的变量称为全局变量. 全局变量作用域是整个程序,局部变量作用域是定义该变量的子程序 ...

  4. (八)Activiti之流程变量和局部流程变量

    一.流程变量 1.1 概念 如果,当流程走到"学生请假"这个任务节点的时候,此时可以用TaskService设置流程变量,变量值包含请假人.请假时间.请假理由等信息,这些信息存在表 ...

  5. test dword ptr [eax],eax ; probe page.局部数组变量定义所分配的最大空间为1M

    问题的出现 使用VS2017编写程序时,程序编译可以通过,但运行时就会弹出错误 经过查证发现: 这跟局部数组变量定义所分配的最大空间设置大小有关. 局部变量的申请空间是存放于栈中,windows里默认 ...

  6. 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 ...

  7. Java基础-内部类-为什么局部和匿名内部类只能访问局部final变量

    先看下面这段代码: public class Test { public static void main(String[] args) { } public void test(final int ...

  8. [ python ] 全局和局部作用域变量的引用

    全局与局部变量的引用 (a)locals(b)globals 这里还需要在补充2个关键字一起比较学习,关键字:(c)nonlocal(d)global locals 和 globals locals: ...

  9. C++函数返回局部指针变量

    遇到过好几次关于函数返回指针变量问题,有时候是可以的,有时候是不可以的,然后就混乱了.今天研究了下,结果发现原来和内存分配有关. 用下面的例子分析下吧: char * test() { char a[ ...

随机推荐

  1. [译]Vulkan教程(27)Image

    [译]Vulkan教程(27)Image Images Introduction 入门 The geometry has been colored using per-vertex colors so ...

  2. java自学-数组

    1.数组是什么 前边说过java的基本数据类型,数组,就是装这些基本类型的容器.每个基本类型的变量都是单个的,数组就是这些单个元素的组合. 2.创建数组 方式一 格式: 数组存储的数据类型[] 数组名 ...

  3. 将vue项目部署在Linux的Nginx下,并设置为局域网内访问

    1. 下载 wget http://nginx.org/download/nginx-1.12.2.tar.gz 2. 解压缩 tar -zxvf linux-nginx-1.12.2.tar.gz ...

  4. 使用fidder对安卓模拟器进行抓包

    本文主要介绍fiddler4对安卓模拟器的抓包设置 首先在官网下载fiddler4进行安装,然后下载安卓模拟器,这里使用网易的MuMu模拟器. 1.fiddler4的设置对于fiddler4的设置,首 ...

  5. IMP-00009: abnormal end of export file解决方案

    一.概述 最近在测试环境的一个oracle数据库上面,使用exp将表导出没有问题,而将导出的文件使用imp导入时却出现了如下错误. IMP-00009: abnormal end of export ...

  6. C++ `endl` 与 `\n` 的区别

    std::cout << std::endl : 插入换行并刷新缓存区 (flush the buffer) std::cout << "\n" : 插入换 ...

  7. CAD转DXF怎么转换?教你三种转换方法

    CAD图纸在我们日常生活中都是可见到的,因为CAD图纸文件的格式是多样的,在工作中就需要经常将CAD的格式进行转换.那CAD转DXF怎么转换呢?这个问题很多的小伙伴们都遇到过,下面小编就来教大家三种转 ...

  8. .netcore 中使用开源的AOP框架 AspectCore

    AspectCore Project 介绍 什么是AspectCore Project ? AspectCore Project 是适用于Asp.Net Core 平台的轻量级 Aop(Aspect- ...

  9. DBUtils框架的使用(上)

    昨天做了这么多的铺垫,当然就是为了引出今天的DBUtils框架了,它的实现原理跟我们编写的简易框架是类似的. 话不多说,进入正题. commons-dbutils 是 Apache 组织提供的一个开源 ...

  10. C#后台架构师成长之路-高阶知识体系核心

    了解了这些东西,熟悉了运用基本都是高工级别的了,其他修修补补就行了.... 1.三种预定义特性:attributeUsage.Conditional.obsolete,允许创建自定义特性,派生自Sys ...