C#闭包陷阱

在C#中,lambda(匿名委托)使用时,编译器会自动生成一个类来保存lambda中的方法以及字段,当lambda引用外部变量的时候,编译器会在生成的类中创建一个字段保存引用到的变量。
在for循环定义的循环变量是for块级别的,没有在循环迭代时候创建新的变量。也就是说想在for循环中的lambda执行的时候调用外部变量将只能获取最新的值,所以我们需要在for循环中创建一个临时变量这样lambda将这个临时变量的值存起来。

用例:

class Program
    {
        static void Main(string[] args)
        {
            List<Person> li = new List<Person>()
            {
                new Person("kk"),
                new Person("cc"),
            };
            Action[] array = new Action[2];
            for (int i = 0; i < 2; i++)
            {
                array[i] = () => { Console.WriteLine(li[i].name);};
            }
            for (int i = 0; i < 2; i++)
            {
                array[i]();
            }
        }
    }
    public class Person
    {
        public string name;

        public Person()
        {
        }

        public Person(string name)
        {
            this.name = name;
        }
    }

反编译
反编译可以看到在Main中有一个<>c__DisplayClass0_2.i,此变量会在复制后变为赋值时循环的次数加一

private static void Main(string[] args)
        {
            Program.<>c__DisplayClass0_0 <>c__DisplayClass0_ = new Program.<>c__DisplayClass0_0();
            <>c__DisplayClass0_.li = new List<Person>
            {
                new Person("kk"),
                new Person("cc")
            };
            Action[] array = new Action[2];
            Program.<>c__DisplayClass0_1 <>c__DisplayClass0_2 = new Program.<>c__DisplayClass0_1();
            <>c__DisplayClass0_2.CS$<>8__locals1 = <>c__DisplayClass0_;
            <>c__DisplayClass0_2.i = 0;
            while (<>c__DisplayClass0_2.i < 2)
            {
                array[<>c__DisplayClass0_2.i] = new Action(<>c__DisplayClass0_2.<Main>b__0);
                int i = <>c__DisplayClass0_2.i;
                <>c__DisplayClass0_2.i = i + 1;
            }
            for (int j = 0; j < 2; j++)
            {
                array[j]();
            }
        }

修改后:

static void Main(string[] args)
        {
            List<Person> li = new List<Person>()
            {
                new Person("kk"),
                new Person("cc"),
            };
            Action[] array = new Action[2];
            for (int i = 0; i < 2; i++)
            {
                int index = i;
                array[index] = () => { Console.WriteLine(li[index].name);};
            }
            for (int i = 0; i < 2; i++)
            {
                array[i]();
            }
        }

反编译:

private static void Main(string[] args)
        {
            Program.<>c__DisplayClass0_0 <>c__DisplayClass0_ = new Program.<>c__DisplayClass0_0();
            <>c__DisplayClass0_.li = new List<Person>
            {
                new Person("kk"),
                new Person("cc")
            };
            Action[] array = new Action[2];
            for (int i = 0; i < 2; i++)
            {
                Program.<>c__DisplayClass0_1 <>c__DisplayClass0_2 = new Program.<>c__DisplayClass0_1();
                <>c__DisplayClass0_2.CS$<>8__locals1 = <>c__DisplayClass0_;
                <>c__DisplayClass0_2.index = i;
                array[<>c__DisplayClass0_2.index] = new Action(<>c__DisplayClass0_2.<Main>b__0);
            }
            for (int j = 0; j < 2; j++)
            {
                array[j]();
            }
        }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,679评论 1 32
  • 在C语言中,五种基本数据类型存储空间长度的排列顺序是: A)char B)char=int<=float C)ch...
    夏天再来阅读 4,066评论 0 2
  • 指针是C语言中广泛使用的一种数据类型。 运用指针编程是C语言最主要的风格之一。利用指针变量可以表示各种数据结构; ...
    朱森阅读 3,616评论 3 44
  • 官网 中文版本 好的网站 Content-type: text/htmlBASH Section: User ...
    不排版阅读 4,724评论 0 5
  • 部分图片会上传失败,为了良好的阅读体验建议查看github原文 1.DNS解析 我们在浏览器上输入地址时,如www...
    mytac阅读 1,698评论 11 65

友情链接更多精彩内容