所以我创建了一个字典的投影,其中包含我想删除的项目。
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/