2

对于你们中的一些人来说,这可能看起来微不足道,但我对下面的这两个例子感到困惑。

int[] numbers = new int[] { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
int i = 0;

var simpleQuery =
    from num in numbers
    select ++i;

foreach (var item in simpleQuery)
{
    Console.WriteLine("v = {0}, i = {1}", item, i); // now i is incremented          
}

输出:

v = 1, i = 1
v = 2, i = 2
v = 3, i = 3
v = 4, i = 4
v = 5, i = 5
v = 6, i = 6
v = 7, i = 7
v = 8, i = 8
v = 9, i = 9
v = 10, i = 10

它更新了 i 的值,到目前为止一切都很好。但是当我尝试更新数组的元素时,它就不起作用了。

int[] numbers = new int[] { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
var simpleQuery =
    from num in numbers
    select ++num;

int i = 0;
foreach (var item in simpleQuery)
{
    Console.WriteLine("v = {0}, num = {1}", item, numbers[i++]); // now num is NOT incremented???
}

输出:

v = 6, num = 5
v = 5, num = 4
v = 2, num = 1
v = 4, num = 3
v = 10, num = 9
v = 9, num = 8
v = 7, num = 6
v = 8, num = 7
v = 3, num = 2
v = 1, num = 0

这背后的原因可能是什么?

编辑: 我认为第二个示例将输出:

v = 6, num = 6
v = 5, num = 5
v = 2, num = 2
v = 4, num = 4
v = 10, num = 10
v = 9, num = 9
v = 7, num = 7
v = 8, num = 8
v = 3, num = 3
v = 1, num = 1
4

3 回答 3

3

您的查询实际上看起来像

numbers.Select(num => ++num)

这实际上是调用扩展方法

Enumerable.Select(numbers, new Func<int, int>(num => ++num))

此方法对数组的每个项目执行选择器。Selector 是一个匿名函数(即这个函数的名称将由编译器生成)。每个项目都传递给该函数。这就是为什么数组中的项目保持不变的原因——整数是一种值类型。值类型是按值传递的(即创建项目的副本,而不是传递对项目的引用)。因此,内部选择器副本被修改并返回。这不会影响数组中的原始项目。

在第一种情况下,您在委托中捕获了对i变量的引用,这就是i更改的原因:

Enumerable.Select(numbers, new Func<int, int>(num => ++i))

数组项仍然作为方法参数传递到这里,但i在方法体中被捕获。实际上,这些变量(它们是方法体的一部分)被编译器替换 - 在您的第一种情况下,变量i被移动到类的字段,并且对该变量的所有调用都被替换为对类字段的调用。即在第一种情况下编译的代码将如下所示:

Foo foo = new Foo();
foreach(int num in numbers)
   Console.WriteLine(foo.Bar(num));

whereFoo是生成的嵌套类,并且Bar是选择器委托,它是在该类中生成的方法。

private class Foo
{
   private int _i; // variable is captured by delegate

   public int Bar(int x)
   {
       _i = _i + 1; // thats why it has new value on next call
       return _i;
   }
}
于 2013-07-02T07:05:23.883 回答
1

您的第一个查询对引用进行操作i(作为闭包的一部分捕获),而您的第二个查询对每个数组元素的副本进行操作,而不是引用。传递给 LINQ 查询的表达式映射到 lambda 表达式,输入到 lambda 表达式(例如num在您的示例中)的变量是值类型的副本,而不是引用。

于 2013-07-02T07:21:11.827 回答
0
var simpleQuery =
from num in numbers
select ++i;

方法:

for(int i=0; i<numbers.Length; i++)
{
    simpleQuery.Add(++i);
}

var simpleQuery =
from num in numbers
select ++num;

方法:

for(int i=0; i<numbers.Length; i++)
{
    simpleQuery.Add(numbers[i] + 1);
}
于 2013-07-02T06:01:45.860 回答