c# - 使用 Linq 进行复制/过滤会在 foreach 循环中产生动态结果吗?

标签 c# linq foreach

所以我创建了一个字典的投影,其中包含我想删除的项目。

var toRemoveList =
    this.outputDic.Keys.Where(key =>
        this.removeDic.ContainsKey(key));

然后我遍历从实际字典中删除的结果

        foreach(var key in toRemoveList)
            this.outputDic.Remove(key);

然而,在 foreach 期间抛出一个异常,指出列表在循环期间被修改。但是,怎么会呢? linq 查询是否有点动态并且每次字典更改时都会重新评估?查询末尾的一个简单的 .ToArray() 调用解决了问题,但我认为,它甚至不应该出现在第一位。

最佳答案

So I create this projection of a dictionary of items I would like to remove.

var toRemoveList =
  this.outputDic.Keys.Where(key =>
    this.removeDic.ContainsKey(key));

正如我常说的,如果我可以教人们关于 LINQ 的一件事,那就是:查询表达式的结果是查询,而不是执行查询的结果。你现在有一个对象,意思是“字典的键,键是……某物”。它不是该查询的结果,而是该查询。查询本身就是一个对象;在您要求之前,它不会为您提供结果集。

然后你这样做:

    foreach(var key in toRemoveList)
        this.outputDic.Remove(key);

那你在干什么?您正在遍历查询。迭代查询执行查询,因此查询 迭代原始字典。但是你随后从字典中删除了一个项目,当你迭代它时,这是非法的。

imo, it shouldn't even occur in the first place.

您对世界应该怎样的看法很普遍,但按您的方式行事会导致严重的低效。让我们假设创建查询会立即执行查询,而不是创建查询对象。这是做什么的?

var query = expensiveRemoteDatabase
    .Where(somefilter)
    .Where(someotherfilter)
    .OrderBy(something);

第一次调用 Where 会产生一个查询,然后在您的世界中执行,从远程数据库中提取与该查询匹配的所有记录。第二次调用 Where 然后说“哦,抱歉,我也想在这里应用这个过滤器,我们可以再次执行整个查询吗,这次使用第二个过滤器?”然后 计算整个记录集,然后我们说“哦,不,等等,我忘了告诉你当你构建最后一个查询对象时,我们需要对其进行排序,那么数据库,你能帮我第三次运行这个查询吗?”

现在您也许明白为什么查询会产生一个一个查询,但在需要时才会执行?

关于c# - 使用 Linq 进行复制/过滤会在 foreach 循环中产生动态结果吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18022755/

相关文章:

c# - 在显示之前设置数组的最大和 C#

c# - Need Insight - 需要 webservice c# .net TO php mysql 的软件开发人员

.net - 当我们对子列表应用条件时,如何过滤父列表<>?

javascript - javascript forEach() 函数如何在 mongodb shell 中迭代文档?

PHP数组切片多个数组不起作用

mysql - Expressjs : Open view after foreach with mysql result.

c# - 使用 C# Cheat Engine 中的指针

c# - 在 C# 中从匿名类型生成类

c# - Linq - 如何进行两个选择查询?

c# - 在linq c#中删除多行