简体   繁体   English

Linq IQueryable通用过滤器

[英]Linq IQueryable Generic Filter

I am looking for a generic Filter for the searchText in the query of any Column/Field mapping 我正在寻找任何列/字段映射的查询中的searchText的通用过滤器

public static IQueryable<T> Filter<T>(this IQueryable<T> source, string searchTerm)
{
   var propNames = typeof(T).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(e=>e.PropertyType == typeof(String)).Select(x => x.Name).ToArray();


 //I am getting the property names but How can I create Expression for

 source.Where(Expression)
}

Here I am giving you an example scenario 在这里,我给你一个示例场景

Now From my HTML5 Table in Asp.net MVC4 , I have provided a Search box to filter results of entered text , which can match any of the below columns/ Menu class Property values , and I want to do this search in Server side , how can I implement it. 现在从我在Asp.net MVC4中的HTML5表中,我提供了一个搜索框来过滤输入文本的结果,它可以匹配以下任何列/菜单类属性值,我想在服务器端执行此搜索,如何我可以实现它吗?

EF Model Class EF模型类

 public partial class Menu
    {
        public int Id { get; set; }
        public string MenuText { get; set; }
        public string ActionName { get; set; }
        public string ControllerName { get; set; }
        public string Icon { get; set; }
        public string ToolTip { get; set; }
        public int RoleId { get; set; }

        public virtual Role Role { get; set; }
    }

You can use Expressions: 您可以使用表达式:

private static Expression<Func<T, bool>> GetColumnEquality<T>(string property, string term)
{
    var obj = Expression.Parameter(typeof(T), "obj");        

    var objProperty = Expression.PropertyOrField(obj, property);
    var objEquality = Expression.Equal(objProperty, Expression.Constant(term));

    var lambda = Expression.Lambda<Func<T, bool>>(objEquality, obj);

    return lambda;
}

public static IQueryable<T> Filter<T>(IQueryable<T> source, string searchTerm)
{
    var propNames = typeof(T).GetProperties(BindingFlags.Instance | BindingFlags.Public)
                             .Where(e => e.PropertyType == typeof(string))
                             .Select(x => x.Name).ToList();

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

    foreach(var name in propNames)
    {
        predicate = predicate.Or(GetColumnEquality<T>(name, searchTerm));
    }

    return source.Where(predicate);
}

Combined with the name PredicateBuilder from C# in a NutShell. 结合NutShell中来自C#的名称PredicateBuilder Which is also a part of LinqKit . 这也是LinqKit的一部分。

Example: 例:

public class Foo
{
    public string Bar { get; set; }
    public string Qux { get; set; }
}

Filter<Foo>(Enumerable.Empty<Foo>().AsQueryable(), "Hello");

// Expression Generated by Predicate Builder
// f => ((False OrElse Invoke(obj => (obj.Bar == "Hello"), f)) OrElse Invoke(obj => (obj.Qux == "Hello"), f))
void Main()
{
   // creates a clause like 
   // select * from Menu where MenuText like '%ASD%' or ActionName like '%ASD%' or....
    var items = Menu.Filter("ASD").ToList();
}

// Define other methods and classes here
public static class QueryExtensions
{
    public static IQueryable<T> Filter<T>(this IQueryable<T> query, string search)    
    {           
        var properties = typeof(T).GetProperties().Where(p => 
                /*p.GetCustomAttributes(typeof(System.Data.Objects.DataClasses.EdmScalarPropertyAttribute),true).Any() && */
                p.PropertyType == typeof(String));        

        var predicate = PredicateBuilder.False<T>();
        foreach (var property in properties )
        {
           predicate = predicate.Or(CreateLike<T>(property,search));
        }
        return query.AsExpandable().Where(predicate);
    }
    private static Expression<Func<T,bool>> CreateLike<T>( PropertyInfo prop, string value)
    {       
        var parameter = Expression.Parameter(typeof(T), "f");
        var propertyAccess = Expression.MakeMemberAccess(parameter, prop);                    
        var like = Expression.Call(propertyAccess, "Contains", null, Expression.Constant(value,typeof(string)));

        return Expression.Lambda<Func<T, bool>>(like, parameter);       
    }

}

You need to add reference to LinqKit to use PredicateBuilder and AsExpandable method otherwise won't work with EF, only with Linq to SQL 您需要添加对LinqKit的引用以使用PredicateBuilder和AsExpandable方法,否则将无法使用EF,仅使用Linq to SQL

If you want Col1 like '%ASD%' AND Col2 like '%ASD%' et , change PredicateBuilder.False to PredicateBuilder.True and predicate.Or to predicate.And 如果你想让Col1 like '%ASD%' AND Col2 like '%ASD%' etPredicateBuilder.False PredicateBuilder.True更改为PredicateBuilder.True并将predicate.Or 。或者更改为predicate.And

Also you need to find a way to distinguish mapped properties by your own custom properties (defined in partial classes for example) 您还需要找到一种方法来区分您自己的自定义属性(例如在部分类中定义)的映射属性

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM