Как создать предикат динамически

Привет, я хочу создать список на основе строки поиска, используя выражения предиката.

У меня список типов продуктов содержит разные названия.

List<products> list1 = new List<products>();

        list1.Add(new products("sowmya"));
        list1.Add(new products("Jane"));
        list1.Add(new products("John"));
        list1.Add(new products("kumar"));
        list1.Add(new products("ramya"));
        listBox1.ItemsSource = list1;

Теперь я хочу отфильтровать содержимое на основе пользовательского ввода. Пользователь введет n строк с «+» в качестве разделителя. После получения строк я передам их для предикатного объекта, подобного этому

 private void textBox1_KeyDown(object sender, KeyEventArgs e)
    {
        List<products> list2 = new List<products>();
        Expression<Func<products, bool>> predicate = PredicateBuilder.True<products>();
        if (e.Key == Key.Enter)
        {
            string Searchstring = textBox1.Text.ToString().Trim();
            string[] separator = new string[] { "+" };
            string[] SearchItems=Searchstring.Split(separator,StringSplitOptions.None);
            foreach (string str in SearchItems)
            {
                string temp = str;
                predicate =p => p.Name.Contains(temp.ToLower());                   
            }

            list2 = list1.AsQueryable().Where(predicate).ToList();
            listBox1.ItemsSource = list2;
        }
    }

Если я ввожу более одной строки (sowmya + jane + john), она дает только результат последней строки (john), но мне нужен список всех совпадающих строк.

Пожалуйста, ответьте на этот вопрос, потому что я пытаюсь это сделать, но не могу получить результат.

Пожалуйста, помогите, спасибо.


person Sowmya    schedule 17.06.2011    source источник
comment
dup: stackoverflow.com/questions/845059/   -  person Jahan    schedule 17.06.2011


Ответы (3)


Инициализировать предикат как false

Expression<Func<products, bool>> predicate = PredicateBuilder.False<products>();

Вам нужно объединить предикаты, используя Or

foreach (string str in SearchItems)
{
    string temp = str;
    predicate = predicate.Or(p => p.NameToLower().Contains(temp.ToLower()));                   
}

Источник для построителя предикатов здесь. Он является частью LINQKit.

Код, если ссылка идет

using System;
using System.Linq;
using System.Linq.Expressions;
using System.Collections.Generic;

public static class PredicateBuilder
{
  public static Expression<Func<T, bool>> True<T> ()  { return f => true;  }
  public static Expression<Func<T, bool>> False<T> () { return f => false; }

  public static Expression<Func<T, bool>> Or<T> (this Expression<Func<T, bool>> expr1,
                                                      Expression<Func<T, bool>> expr2)
  {
    var invokedExpr = Expression.Invoke (expr2, expr1.Parameters.Cast<Expression> ());
    return Expression.Lambda<Func<T, bool>>
          (Expression.OrElse (expr1.Body, invokedExpr), expr1.Parameters);
  }

  public static Expression<Func<T, bool>> And<T> (this Expression<Func<T, bool>> expr1,
                                                       Expression<Func<T, bool>> expr2)
  {
    var invokedExpr = Expression.Invoke (expr2, expr1.Parameters.Cast<Expression> ());
    return Expression.Lambda<Func<T, bool>>
          (Expression.AndAlso (expr1.Body, invokedExpr), expr1.Parameters);
  }
}
person Eranga    schedule 17.06.2011
comment
Если я использую And для объединения предикатов, он дает все элементы списка, но не совпадающий результат строки - person Sowmya; 17.06.2011
comment
@Sowmya обновил мой ответ. вам нужно использовать temp, а не str внутри предиката - person Eranga; 17.06.2011
comment
Эй, Эранга, большое спасибо, все работает нормально. На самом деле ты мне помог - person Sowmya; 17.06.2011
comment
Это predicatebuilder то же самое, что и в этой статье albahari.com/nutshell/predicatebuilder.aspx? - person wtf512; 04.04.2017

Здесь не нужно создавать предикат. Вы можете попробовать что-то вроде этого

List<products> list1 = new List<products>();

list1.Add(new products("sowmya"));
list1.Add(new products("Jane"));
list1.Add(new products("John"));
list1.Add(new products("kumar"));
list1.Add(new products("ramya"));

string input = "aaa+kuma+ram";
List<string> searchStrings =
    input.Split(new string[] { "+" }, StringSplitOptions.None)
    .Select(s => s.ToLower())
    .ToList();

List<products> list2 = (
    from p in list1
    where searchStrings.Any(s => p.Name.Contains(s))
    select p).ToList();

list2 будет содержать «кумар» и «рамья».

person Gebb    schedule 17.06.2011
comment
Спасибо, и это будет работать и работать, я знал это, но мне нужно построить его, используя только предикат, пожалуйста, помогите мне. - person Sowmya; 17.06.2011
comment
@Sowmya, тогда тебе следует попробовать то, что предлагает Эранга. - person Gebb; 17.06.2011

Поскольку я не уверен, что у экземпляра предиката есть метод And, я предлагаю вам использовать этот код:

var list = list1.AsQueryable();
foreach (string str in SearchItems)
{
     list = list.Where(p => p.Name.ToLower().Contains(str.ToLower()));
}
listBox1.ItemsSource = list.ToList();
person Maziar Taheri    schedule 17.06.2011
comment
если я использую этот код, он дает только одну строку результата, но если я ввожу две строки, это не дает никакого результата. Пожалуйста, найдите решение. Мне это нужно. - person Sowmya; 17.06.2011
comment
это может быть из-за первого .ToLower(), который я только что добавил. если нет, убедитесь, что ваше свойство Name действительно содержит обе строки, которые вы ищете - person Maziar Taheri; 17.06.2011
comment
та же проблема снова дает только последний результат строки. Я уверен, что свойство Name содержит обе строки - person Sowmya; 17.06.2011
comment
вы должны проверить, правильно ли вы скопировали и вставили код. - person Maziar Taheri; 17.06.2011