3

我对 C# 了解很多,但这个让我很困惑,而谷歌也没有帮助。

我有一个 IEnumerable 范围的对象。我想在第一个上设置一个属性。我这样做了,但是当我在修改后枚举对象范围时,我看不到我的变化。

这是一个很好的问题示例:

    public static void GenericCollectionModifier()
    {
        // 1, 2, 3, 4... 10
        var range = Enumerable.Range(1, 10);

        // Convert range into SubItem classes
        var items = range.Select(i => new SubItem() {Name = "foo", MagicNumber = i});

        Write(items);  // Expect to output 1,2,3,4,5,6,7,8,9,10

        // Make a change
        items.First().MagicNumber = 42;

        Write(items);  // Expect to output 42,2,3,4,5,6,7,8,9,10
        // Actual output: 1,2,3,4,5,6,7,8,9,10
    }

    public static void Write(IEnumerable<SubItem> items)
    {
        Console.WriteLine(string.Join(", ", items.Select(item => item.MagicNumber.ToString()).ToArray()));
    }

    public class SubItem
    {
       public string Name;
       public int MagicNumber;
    }

C# 的哪个方面阻止了我的“MagicNumber = 42”更改输出?有没有一种方法可以让我的更改“坚持”而无需进行一些时髦的转换为 List<> 或数组?

谢谢!-麦克风

4

5 回答 5

8

当您调用 First() 时,它会枚举这段代码的结果:

Select(i => new SubItem() {Name = "foo", MagicNumber = i});

请注意, Select 是一个惰性枚举器,这意味着它仅在您向其中请求项目时才进行选择(并且每次您询问时都会这样做)。结果不会存储在任何地方,因此当您调用 items.First() 时,您会得到一个新SubItem实例。然后,当您将项目传递给 Write 时,它​​会获得一大堆新SubItem实例 - 而不是您之前获得的那个。

如果要存储选择的结果并对其进行修改,则需要执行以下操作:

var items = range.Select(i => new SubItem() {Name = "foo", MagicNumber = i}).ToList();
于 2008-11-25T14:35:44.577 回答
0

我怀疑在后台发生了什么事。很可能是因为 IEnumerables 只能迭代一次。

如果在分配给“项目”时在调用 Select() 之后添加“ToList()”,它是否有效?

于 2008-11-25T14:34:15.687 回答
0

我唯一能想到的是 items.First() 将 SubItem 的副本传递给您而不是引用,因此当您设置它时,不会进行更改。

我不得不假设它与 IQueryable 只能迭代一次有关。您可能想尝试更改此设置:

// Convert range into SubItem classes
var items = range.Select(i => new SubItem() {Name = "foo", MagicNumber = i});

// Convert range into SubItem classes
var items = range.Select(i => new SubItem() {Name = "foo", MagicNumber = i}).ToList();

看看有没有不同的结果。

于 2008-11-25T14:34:36.823 回答
0

您不能/不应该通过枚举器修改集合。我很惊讶这不会引发异常。

于 2008-11-25T14:39:34.677 回答
0

.First() 是一种方法,而不是属性。它在 Enumerable 的第一个位置返回对象的新实例。

于 2008-11-25T14:40:48.123 回答