6

我正在尝试遍历字符串数组并动态创建IQueryable查询。它非常简单,但这是我卡住的地方

var query = context.QuestionsMetaDatas.AsQueryable();

var keywords=new List<string>(){ "Test1","Test2" };

foreach(var key in keywords)
{
   query=query.Where(a=>a.Text.Contains(key));
}

现在的问题是,当查询生成时,它的编译为

select * from QuestionsMetaDatas where Text Like "Test1" AND Text Like "Test2"

而不是AND我希望生成查询OR......现在我该如何实现呢?

4

3 回答 3

6

我使用了 Raphael 建议的谓词生成器,它只是包含在您的项目中的一个文件,然后您的示例变为:

var keywords=new List<string>(){ "Test1","Test2" };

var predicate = PredicateBuilder.False<QuestionsMetaDatas>();

foreach (var key in keywords)
{
  predicate = predicate.Or (a => a.Text.Contains (key));
}

var query = context.QuestionsMetaDatas.AsQueryable().Where(predicate);

生成您正在寻找的 OR 查询。

于 2013-09-12T20:32:16.237 回答
5

您是否尝试过包含其他方式?

var keywords=new List<int>(){ "Test1","Test2" };
query=query.Where(a=>keywords.Contains(a));

这就像一个IN子句

于 2013-09-12T12:23:50.210 回答
2

您可以查看谓词生成器,或构建自己的表达式(这里是一个带有静态扩展方法的可能解决方案 on IQueryable<QuestionsMetadatas>

public static IQueryable<QuestionsMetaDatas> FilterText(this IQueryable<QuestionsMetaDatas> queryable, IEnumerable<string> keywords)
        {
            var entityType = typeof(QuestionsMetaDatas);
            var parameter = Expression.Parameter(entityType, "a");
            var containsMethod = typeof(string).GetMethod("Contains"
                                                           , new[] { typeof(string) });
            var propertyExpression = Expression.Property(parameter, "Text");
            Expression body = Expression.Constant(false);
            foreach (var keyword in keywords)
            {
                var innerExpression = Expression.Call(propertyExpression, containsMethod, Expression.Constant(keyword));
                body = Expression.OrElse(body, innerExpression);
            }
            var lambda = Expression.Lambda<Func<QuestionsMetaDatas, bool>>(body, new[] { parameter });
            return queryable.Where(lambda);

        }

哪里lambda看起来像这样:

a => ((False OrElse a.Text.Contains("firstKeyWord")) OrElse a.Text.Contains("secondKeyWord"))

和用法是

var query = context.QuestionsMetaDatas.AsQueryable();

var keywords=new List<string>(){ "Test1","Test2" };

query = query.FilterText(keywords);

或更短

var query = context.QuestionsMetaDatas.AsQueryable().FilterText(new[]{"Test1", "Test2"});
于 2013-09-12T15:19:41.590 回答