3

我正在尝试创建一个扩展“WhereNot”

所以我可以使用:

Dim x = "Hello world "
Dim y = x.Split.WhereNot(AddressOf String.IsNullOrEmpty)

请注意,我的目标是学习 linq 表达式;不能解决我的问题。

我创建了这个功能:

 <Extension()> _
 Public Function WhereNot(Of TElement)(ByVal source As IQueryable(Of TElement), ByVal selector As Expression(Of Func(Of TElement, Boolean))) As IQueryable(Of TElement)
  Return source.Where(GetWhereNotExpression(selector))
 End Function

我不知道如何切换布尔标志,函数 Negate 会做吗?

欢迎使用 vb.net 和 C# 中的答案

4

2 回答 2

12

我意识到这是一个非常古老的已回答问题,但我认为所选答案具有误导性,因为提问者正在寻找一个表达式,而所选答案提供了一个 lambda。

这意味着它从 IQueryable 调用将返回一个IEnumerable<T>,而不是一个IQueryable<T>。这会强制表达式进行编译,并可能导致对 Linq 提供程序的访问优化不佳。

这是一个直接解决原始问题的答案。

public static class NegationExpressionHelper
{
    public static IQueryable<T> WhereNot<T>(this IQueryable<T> queryable, Expression<Func<T,bool>> predicate)
    {
        return queryable.Where(predicate.Invert());
    }
    public static Expression<Func<T, bool>> Invert<T>(this Expression<Func<T, bool>> original)
    {
        return Expression.Lambda<Func<T, bool>>(Expression.Not(original.Body), original.Parameters.Single());
    }
}
于 2010-12-19T07:19:44.017 回答
1

是的,像这样的 Negate 方法将帮助您:

Public Function Negate(Of T)(ByVal predicate As Func(Of T, Boolean)) As Func(Of T, Boolean)
   Return Function(x) Not predicate(x)
End Function

然后像这样使用它:

Dim x = "Hello world "
Dim y = x.Split.Where(Negate(Of String)(AddressOf String.IsNullOrEmpty))

或者使用这样的 WhereNot() 方法:

<Extension> _
Public Shared Function WhereNot(Of T)(ByVal source As IEnumerable(Of T), ByVal predicate As Func(Of T, Boolean)) As IEnumerable(Of T)
    Return source.Where(Negate(predicate))
End Function
于 2010-02-15T05:01:10.057 回答