C#闭包陷阱
2019-05-31 本文已影响0人
祝你万事顺利
在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]();
}
}