我正在使用我自己的IQueryable扩展方法<>创建可链接的查询,如FindAll()。FindInZip(12345).NameStartsWith(“XYZ”)。OrderByHowIWantIt()等然后在延迟执行时根据我的扩展方法链创建一个查询。
但问题是,扩展链中的所有位置(FindXYZ,FindInZip等)将始终组合为AND,这意味着我无法做到这样的事情:
FindAll()。FirstNameStartsWith(“X”)。OrLastNameStartsWith(“Z”)因为我不知道如何在一个单独的Where方法中注入OR。
知道如何解决这个问题吗?
额外; 到目前为止,我理解如何将表达式链接为或者如果我将它们包装起来(例如CompileAsOr(FirstNameStartsWith(“A”)。LastNameStartsWith(“Z”)。OrderBy(..))
我试图做的虽然稍微复杂一些(并且PredicateBuilder在这里没有帮助......)因为我希望以后的IQueryable能够基本上访问之前建立的Where条件,而不必将它们包装起来创建他们之间的或。
当每个扩展方法返回IQueryable&lt;&gt;我知道它应该知道某个地方的查询条件的当前状态,这使我相信应该有一些自动化的方法或在所有先前的Where条件中创建一个Or而不必包装你想要的东西。< / p>
答案 0 :(得分:9)
我假设查询的不同部分仅在运行时已知,即您不能仅在||
中使用where
...
一个懒惰的选项是Concat
- 但这往往导致差的TSQL等;但是,我倾向于编写自定义Expression
s。采用的方法取决于提供者是什么,因为LINQ-to-SQL支持EF的不同选项(例如) - 这在此具有真正的影响(因为你不能在EF中使用子表达式)。你能告诉我们哪个?
这里有一些应该与LINQ-to-SQL一起使用的代码;如果你构建一个数组(或列表,并调用.ToArray()
)的表达式,它应该工作正常;示例是LINQ到对象,但应该仍然有效:
static void Main()
{
var data = (new[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }).AsQueryable();
var predicates = new List<Expression<Func<int, bool>>>();
predicates.Add(i => i % 3 == 0);
predicates.Add(i => i >= 8);
foreach (var item in data.WhereAny(predicates.ToArray()))
{
Console.WriteLine(item);
}
}
public static IQueryable<T> WhereAny<T>(
this IQueryable<T> source,
params Expression<Func<T,bool>>[] predicates)
{
if (source == null) throw new ArgumentNullException("source");
if (predicates == null) throw new ArgumentNullException("predicates");
if (predicates.Length == 0) return source.Where(x => false); // no matches!
if (predicates.Length == 1) return source.Where(predicates[0]); // simple
var param = Expression.Parameter(typeof(T), "x");
Expression body = Expression.Invoke(predicates[0], param);
for (int i = 1; i < predicates.Length; i++)
{
body = Expression.OrElse(body, Expression.Invoke(predicates[i], param));
}
var lambda = Expression.Lambda<Func<T, bool>>(body, param);
return source.Where(lambda);
}
答案 1 :(得分:8)
使用PredicateBuilder<T>
。这可能就是你想要的。
答案 2 :(得分:0)
List<string> fruits =
new List<string> { "apple", "passionfruit", "banana", "mango",
"orange", "blueberry", "grape", "strawberry" };
var query = fruits.AsQueryable();
// Get all strings whose length is less than 6.
query = query.Where(fruit => fruit.Length < 6);
// Hope to get others where length is more than 8. But you can't, they're gone.
query = query.Where(fruit => 1 == 1 || fruit.Length > 8);
foreach (string fruit in query)
Console.WriteLine(fruit);
答案 3 :(得分:0)
在理想情况下,我个人认为||
和&&
运算符将是最简单易读的。但是它不会编译。
运算符'||'不能应用于类型'
的操作数Expression<Func<YourClass,bool>>
'和'Expression<Func<YourClass,bool>>
'
因此,我为此使用了扩展方法。在您的示例中,它看起来像这样:
.Where(FindInZip(12345).Or(NameStartsWith("XYZ")).And(PostedOnOrAfter(DateTime.Now))
。
代替:
.Where(FindInZip(12345) || NameStartsWith("XYZ") && (PostedOnOrAfter(DateTime.Now))
。
表达示例:
private Expression<Func<Post,bool>> PostedOnOrAfter(DateTime cutoffDate)
{
return post => post.PostedOn >= cutoffDate;
};
扩展方法:
public static class PredicateExtensions
{
/// <summary>
/// Begin an expression chain
/// </summary>
/// <typeparam id="T""></typeparam>
/// <param id="value"">Default return value if the chanin is ended early</param>
/// <returns>A lambda expression stub</returns>
public static Expression<Func<T, bool>> Begin<T>(bool value = false)
{
if (value)
return parameter => true; //value cannot be used in place of true/false
return parameter => false;
}
public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> left,
Expression<Func<T, bool>> right)
{
return CombineLambdas(left, right, ExpressionType.AndAlso);
}
public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> left, Expression<Func<T, bool>> right)
{
return CombineLambdas(left, right, ExpressionType.OrElse);
}
#region private
private static Expression<Func<T, bool>> CombineLambdas<T>(this Expression<Func<T, bool>> left,
Expression<Func<T, bool>> right, ExpressionType expressionType)
{
//Remove expressions created with Begin<T>()
if (IsExpressionBodyConstant(left))
return (right);
ParameterExpression p = left.Parameters[0];
SubstituteParameterVisitor visitor = new SubstituteParameterVisitor();
visitor.Sub[right.Parameters[0]] = p;
Expression body = Expression.MakeBinary(expressionType, left.Body, visitor.Visit(right.Body));
return Expression.Lambda<Func<T, bool>>(body, p);
}
private static bool IsExpressionBodyConstant<T>(Expression<Func<T, bool>> left)
{
return left.Body.NodeType == ExpressionType.Constant;
}
internal class SubstituteParameterVisitor : ExpressionVisitor
{
public Dictionary<Expression, Expression> Sub = new Dictionary<Expression, Expression>();
protected override Expression VisitParameter(ParameterExpression node)
{
Expression newValue;
if (Sub.TryGetValue(node, out newValue))
{
return newValue;
}
return node;
}
}
#endregion
}
关于通过扩展表达式进行LINQ查询的非常好的文章。也是我使用的扩展方法的来源。