如何动态创建谓词

本文关键字:创建 谓词 动态 何动态 | 更新日期: 2023-09-27 17:59:18

Hi我想使用谓词表达式基于搜索字符串创建一个列表。

我有一个包含不同名称的类型产品列表。

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)的结果,但我想要所有匹配字符串的列表

请回答这个问题,因为我正在尝试,但没有得到结果。

请帮忙,谢谢。

如何动态创建谓词

将谓词初始化为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);
  }
}

您不必在这里构建谓词。你可以试试这种

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将包含"kumar"answers"ramya"。

由于我不确定谓词实例是否有And方法,我建议您使用以下代码:

var list = list1.AsQueryable();
foreach (string str in SearchItems)
{
     list = list.Where(p => p.Name.ToLower().Contains(str.ToLower()));
}
listBox1.ItemsSource = list.ToList();