LINQ中的谓词
如何在LINQ中的Where谓词中指定条件而不获取空引用exception。 例如,如果q
是IQueryable,我该怎么做:
Expression<Func> predicate = p => !search.CategoryId.HasValue || (search.CategoryId.HasValue && search.CategoryId == p.CategoryId); var q2 = q.Where(predicate);
这里search
是一个对象,它包含可能或不可能设置的搜索条件,如search.CategoryId可能没有设置,但如果是,我想获得由该条件设置的产品。
当我这样做时,我得到空引用exception。
你可以使用null-coalescing运算符 ??
用默认值替换可能的空值。 以下集合尝试匹配search.Category(如果存在)或仅创建“always true”表达式。 这将由任何优秀的Linq查询提供程序(例如LinqToSql)进行优化。
Expression> predicate = (p => (search.CategoryId ?? p.CategoryId) == p.CategoryId)); var q2 = q.Where(predicate);
另一种可能性是使用PredicateBuilder动态组合查询谓词。 这就是我使用类似模式进行搜索的方式:
var predicate = PredicateBuilder.True(); if (search.OrderId)) { predicate = predicate.And(a => SqlMethods.Like(a.OrderID, search.OderID); } // ... var results = q.Where(predicate);
让我们剖析一下这条线:
Expression predicate = p => !search.CategoryId.HasValue || (search.CategoryId.HasValue && search.CategoryId == p.CategoryId) var q2 = q.Where(predicate);
那么我们有多少种方法可以获得null
问题呢?
-
search
(您的“捕获”变量)可以为null
-
p
可以为null,表示列表中null
- 你已经处理了
search.CategoryId
为null
的情况(Nullable
) - 但也许
p.CategoryId
(列表中记录的类别)为null
(Nullable
) – 但是,我不确定这会导致NullReferenceException
-
q
(列表/来源)可以为null
所以:你已经淘汰了5个选项中的1个; 看看其他4个? 还有一种可能性,即问题是由代码中未显示的隐形物引起的; 例如, get
可能是:
public int? CategoryId { get {return innerObject.CategoryId;} }
和innerObject
可以为null
; 如果你消除了其他4个(很容易做到),请将此作为最后的手段。