繁体   English   中英

使用包含 int 列表的枚举属性的方法构建 LINQ Lambda 表达式

[英]Build LINQ Lambda Expression with contains method for enum property with list of int

我已经使用 IQueryable 构建了一些用于排序和过滤通用对象的扩展方法来获取数据,到目前为止它适用于所有类型,但对于枚举,我无法弄清楚如何解决这个问题,所以这里有一些例子。

    public enum ClaimStatusEnum
{
    None = 0,
    Open = 10,
    Submitted = 20,
    Rejected = 30,
    Refunded = 40,
    Closed = 50,
    Received = 60,
    Backorder = 70,
    UnderReview = 80
}

请注意! 这只是枚举的一个例子,但我在执行时不知道枚举类型是什么,所以请注意我必须使用泛型类型。

现在这是一个示例,其中包含具有该枚举类型的属性。

 public class Claim
{
    [Key]
    public int Id { get; set; }
    public DateTime DateCreated { get; set; }      
    public ClaimStatusEnum ClaimStatus { get; set; }

}

现在这里是一个过滤器的例子,它与枚举中的 int 值列表一起传递。

var arr = new int[] { 20, 30, 40 };

现在这个例子中的目标是在 Claim 表中找到我们在 int 的 arr 中具有枚举值的记录。

所以我想构建一个 lambda 表达式,它应该是通用的,而不是对类的类型进行硬编码,因为它可以是任何实体,也应该适用于任何给定的枚举。

所以这是我目前使用的完全相同的东西,但唯一的区别是它的属性是 int 而不是枚举,所以它适用于 int 类型,但是当涉及到枚举类型时,我得到一个错误.

                    ConstantExpression c = Expression.Constant(arr);
                MethodInfo mi = typeof(Enumerable).GetMethods(BindingFlags.Static | BindingFlags.Public)
                               .Where(m => m.Name == "Contains")
                               .Single(m => m.GetParameters().Length == 2)
                               .MakeGenericMethod(typeof(int));
                call = Expression.Call(mi, c, m);

任何人都可以在这里帮助我如何使这项工作适用于枚举类型? 我已经尝试了很多不同的方法,它们都抛出错误。

这是如何将其用作扩展方法。

        public static IQueryable<T> EntitySortAndFilter<T>(this IQueryable<T> data, PageFilter filter, T type)
    {

        foreach (var item in filter.Filter.Filters)
        {
            if (item.Value == null || item.Condition == null || item.Field == null)
                continue;

            ParameterExpression e;
            Expression m, call;
            PropertyInfo prop;

            prop = type.GetType().GetProperty(item.Field, BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance);
            e = Expression.Parameter(type.GetType(), "e");
            m = Expression.MakeMemberAccess(e, prop );

            
            var arr = item.Value;
                ConstantExpression c = Expression.Constant(arr);
                MethodInfo mi = typeof(Enumerable).GetMethods(BindingFlags.Static | BindingFlags.Public)
                               .Where(m => m.Name == "Contains")
                               .Single(m => m.GetParameters().Length == 2)
                               .MakeGenericMethod(typeof(int));
                call = Expression.Call(mi, c, m);

            var lambda = Expression.Lambda<Func<T, bool>>(call, e);
            data = data.Where(lambda);

           


        } 

       return data;  

    }

在调用此扩展程序时,我是这样调用的。

var example = query.EntitySortAndFilter(filter, new Claim());

但同样,它应该适用于我想要的任何实体,这就是为什么它是通用的。

我认为你需要做这样的事情:

if(prop.PropertyType.IsEnum) // if property is enum
{    
    // create expression to cast array values to enum
    var castMi = typeof(Enumerable).GetMethod("Cast").MakeGenericMethod(prop.PropertyType);
    var castArrToEnum = Expression.Call(castMi, Expression.Constant(arr)));
    // make contains for needed type
    mi = typeof(Enumerable).GetMethods(BindingFlags.Static | BindingFlags.Public)
            .Where(m => m.Name == "Contains")
            .Single(m => m.GetParameters().Length == 2)
            .MakeGenericMethod(prop.PropertyType);
    call = Expression.Call(mi, castArrToEnum, m);
}

如果这不起作用,您可以随时尝试其他方式:

var mc = Expression.Convert(m, typeof(int)); 
ConstantExpression c = Expression.Constant(arr); 
MethodInfo mi = typeof(Enumerable).GetMethods(BindingFlags.Static | BindingFlags.Public) 
    .Where(m => m.Name == "Contains") 
    .Single(m => m.GetParameters().Length == 2) 
    .MakeGenericMethod(typeof(int)); 
call = Expression.Call(mi, c, mc);

暂无
暂无

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

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