3

我想创建像这样的 lambda

user => user.Address == address

但不是编译一个,我要返回LambdaExpression

如果 lambda 像这样保持不变

user => user.Age == 50

然后我可以使用这个方法

    public static LambdaExpression PropertyEqual(Type tEntityType, string propertyName, object value)
    {
        // entity => entity.PropName == const
        var itemParameter = Expression.Parameter(tEntityType, "entity");
        return Expression.Lambda
        (
            Expression.Equal
            (
                Expression.Property
                (
                    itemParameter,
                    propertyName
                ),
                Expression.Constant(value) // Tried to replace this with Expression.Parameter or Expression.Variable but no luck
            ),
            new[] { itemParameter }
        );
    }

如何使此方法接受变量address来自 lambda 表达式之外的范围?

var addressPropertyName = "Address";
var address = new Address() {...};
var q = Repo.GetQuery().Where(PropertyEqual(typeof(User), addressPropertyName, address))

编辑:澄清我的问题:如何建立Expression生成第一个 lambda 的权利?

更新:这是不可能的,因为EF 不支持非标量变量

我将 lambda 更改为这里user => user.AddressId == addressId建议的。这只是如何从已知的导航属性中获取FK的问题。AddressIdPropertyInfoAddress

4

1 回答 1

4

你不能动态地生成一个变量的闭包(你不能在它的上下文之外扩展一个变量的生命周期),因为这是编译器的一个技巧(重写你的代码来做到这一点)。

如果您不想要闭包但想要附加参数,则可以向表达式添加附加参数。

你可以

Expression<Func<string>> myExpr = () => address;

现在你有一个围绕你的地址关闭的表达式。现在你只需要结合这两个表达式。

您必须将方法更改为:

public static LambdaExpression PropertyEqual<T>(Type tEntityType, string propertyName, Expression<Func<T>> getValue)
{
   // entity => entity.PropName == const
   var itemParameter = Expression.Parameter(tEntityType, "entity");
   return Expression.Lambda
   (
       Expression.Equal
       (
           Expression.Property
           (
               itemParameter,
               propertyName
           ),
           Expression.Invoke(getValue) // You could directly use getValue.Body instead of Expression.Invoke(getValue)
       ),
       new[] { itemParameter }
   );
}
于 2013-08-01T10:12:17.480 回答