使用辅助方法

时间:2016-09-19 08:57:44

标签: c# lambda expression

我有一个主方法,可以为给定的实体创建基础搜索条件。因此,在将此方法应用于查询之前,我会检查默认值。

E.g。

        if (!string.IsNullOrEmpty(value))
            qry = qry.Where(x => x.PropA.Contains(value));

        if (!string.IsNullOrEmpty(anotherValue))
            qry = qry.Where(x => x.PropB.Contains(anotherValue));

但是,我想重构一下并使用辅助方法,但由于我对表达式的知识和经验有限,我很难完成任务。

我有这个锅炉代码,我相信这说明了我正在努力实现的目标:

    IQueryable<T> Test<T, TV>(IQueryable<T> qry, Expression<Func<T, TV>> prop, TV value)
    {
        if (EqualityComparer<TV>.Default.Equals(value, default(TV)))
            return qry;

        var right = Expression.Constant(value);

        var body = Expression.Equal(prop, right);
        var lambda = Expression.Lambda<Func<T, bool>>(body);

        return qry.Where(lambda);
    }

这应该让我能够像这样打电话:

qry = Test(qry, x=>PropA, value);
qry = Test(qry, x=>PropB, anotherValue);

然而问题是body变量导致BinaryExpression,我对如何从这里继续进行完全无知。

1 个答案:

答案 0 :(得分:3)

您必须将方法转换为Expression,然后将其包含为lambda的主体。

因此,从您的锅炉代码开始,在上述更改后,它应该看起来像

    IQueryable<T> Test<T, TV>(IQueryable<T> qry, Expression<Func<T, TV>> prop, string propertyValue)
    {

        MethodInfo method = typeof(string).GetMethod("Contains", new[] { typeof(string) });
        var someValue = Expression.Constant(propertyValue, typeof(string));
        var body = Expression.Call(prop, method, someValue); // pseudocode, to be refined below

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

        return qry.Where(lambda);
    }

现在让我用字符串访问器

对其进行修改
    static IQueryable<T>  Test<T>(IQueryable<T> qry, string propertyName, string propertyValue)
    {
        var parameterExp = Expression.Parameter(typeof(T), "type");
        var propertyExp = Expression.Property(parameterExp, propertyName);
        MethodInfo method = typeof(string).GetMethod("Contains", new[] { typeof(string) });
        var someValue = Expression.Constant(propertyValue, typeof(string));
        var containsMethodExp = Expression.Call(propertyExp, method, someValue);
        var lambda = Expression.Lambda<Func<T, bool>>(containsMethodExp, parameterExp);
        return qry.Where(lambda);
    }

最后一个简单的用法示例

    class MyClass
    {
        public string Myname { get; set; }
    }

    static void Main(string[] args)
    {
        var check = new MyClass() { Myname = "11 aa 22" };
        var check2 = new MyClass() { Myname = "11 bb 22" };
        var x = new List<MyClass>();
        x.Add(check);
        x.Add(check2);
        var q = x.AsQueryable();
        var qry = Test(q, "Myname", "bb");
    }

好吧,如果您更喜欢属性选择器,那么帮助程序将变为

    static IQueryable<T>  Test<T>(IQueryable<T> qry, Expression<Func<T, string>> selector, string propertyValue)
    {
        var parameterExp = Expression.Parameter(typeof(T), "type");

        var memberExpression = (MemberExpression)selector.Body;
        var parameterTProperty = (PropertyInfo)memberExpression.Member;
        var propertyExp = Expression.Property(parameterExp, parameterTProperty);

        MethodInfo method = typeof(string).GetMethod("Contains", new[] { typeof(string) });
        var someValue = Expression.Constant(propertyValue, typeof(string));
        var containsMethodExp = Expression.Call(propertyExp, method, someValue);
        var lambda = Expression.Lambda<Func<T, bool>>(containsMethodExp, parameterExp);
        return qry.Where(lambda);
    }

用作

        var qry = Test(q, z => z.Myname , "bb");