LINQ to SQL PredicateBuilder

时间:2009-10-27 01:39:55

标签: linq-to-sql predicatebuilder

我正在使用这里看到的PredicateBuilder http://www.albahari.com/nutshell/predicatebuilder.aspx,一切都很好,现在我可以生成动态LINQ到SQL表达式,但我不明白的是为什么当我在这样的循环上时:< / p>

var inner = PredicateBuilder.False<MyType>();
foreach (var f in Filtermodel.InstrumentsFilterList.Where(s => s.isActive))
        {
          int temp = f.InstrumentID;
          inner = inner.Or(ud => ud.InstrumentId == temp);
        }

为什么我必须使用那个临时变量?,我尝试使用“f”迭代器变量,但它只获取列表中每个迭代的最后一个值,就像它通过引用传递一样...

2 个答案:

答案 0 :(得分:10)

因为PredicateBuilder正在构建一个表达式,该表达式将在稍后的时间点执行。当编译器为委托生成闭包时,它会找到在当前作用域中创建的任何值,并将它们也带入闭包中。由于InstrumentID是一个值类型(int),初始化和复制值意味着每个委托/闭包都将携带该值。如果每次都不创建值的副本,则表达式将只具有对f.InstrumentID的文字引用,而不是其基础值。所以稍后,当实际执行表达式时,将对f.InstrumentID进行求值,它将按照上次设置的内容出现,这是最后一次迭代。

答案 1 :(得分:2)

因为它不是在评估条件,而是简单地构建表达式。表达式绑定到foreach中定义的变量,该变量在整个循环的执行过程中保留它的引用。使用临时变量重新定义它会强制每个表达式使用不同的变量,这会强制它在每次迭代时使用值引用实例,而不是让所有迭代引用单个引用并且只具有最后一次迭代的值。 / p>