【问题标题】:Expression Equals表达式等于
【发布时间】:2010-09-12 14:47:42
【问题描述】:

所以,我正在尝试找出表达式树。我正在尝试将动态等于添加到 Queryable 中,其中 T 是几个不同的表之一。我首先检查表中是否包含我要过滤的字段。

ParameterExpression param = Expression.Parameter(typeof(TSource), "x");

Expression conversionExpression = Expression.Convert(Expression.Property(param, _sourceProperty), typeof(TList));

Expression<Func<TSource, TList>> propertyExpression = Expression.Lambda<Func<TSource, TList>>(conversionExpression, param);

Expression<Func<TList, TList, bool>> methodExpression = (x, y) => x.Equals(y);

ReadOnlyCollection<ParameterExpression> parameters = propertyExpression.Parameters;

InvocationExpression getFieldPropertyExpression = Expression.Invoke(
                                                propertyExpression,
                                                parameters.Cast<Expression>());

MethodCallExpression methodBody = methodExpression.Body as MethodCallExpression;
MethodCallExpression methodCall = Expression.Call(methodBody.Method, Expression.Constant(equalTo), getFieldPropertyExpression);

Expression<Func<TSource, bool>> equalsStatement = Expression.Lambda<Func<TSource, bool>>(methodCall, parameters);

return source.Where(equalsStatement);

执行此操作时,调用语句中的 MethodInfo 出现问题。它告诉我;

静态方法需要空实例,非静态方法需要非空实例。

我不是表达式树的大师,但我认为我了解我在这里所做的大约 75% 的事情,并且知道我想要实现的目标。 TList 现在是一个坏名字,但我从一个可以很好地生成 In 语句的示例中得到这个。

我真的在这里寻找解释,这样我就可以自己处理代码,或者找到解释我所缺少的内容的解决方案。

编辑:

好的,在经历了一个非常令人沮丧的下午之后,我仍然觉得我完全理解我在看什么,我想我有一个答案。

ParameterExpression sourceObject = Expression.Parameter(typeof(TSource), "x");

Expression<Func<TSource, bool>> check = Expression.Lambda<Func<TSource, bool>>
        (
            Expression.Equal(
            Expression.MakeMemberAccess(sourceObject, typeof(TSource).GetProperty(_sourceProperty)),
            Expression.Constant(equalTo)
        ),
        sourceObject
);

return source.Where(check);

有人可以向我解释为什么原版不适合我想要做的事情吗?我想了解更多关于实际过程的信息,但我觉得我没有像我希望的那样快速掌握它。

【问题讨论】:

    标签: c#-4.0 lambda


    【解决方案1】:

    Expression.Call 有两组重载(每组都有很多重载)。一组用于实例方法,另一组用于静态方法。在那些静态方法中,第一个参数是 MethodInfo 对象——和你的完全一样。对于 instance 方法,第一个参数应该是表示目标的表达式(即方法调用中“.”的左侧。)鉴于您收到的错误,听起来像MethodInfo 表示一个非静态方法,因此您必须提供一个表示实例的表达式作为第一个参数。

    【讨论】:

    • 感谢您的回复。我将此标记为答案,因为它让我以我应该看待问题的方式看待问题。