创建包含外部变量的LambdaExpression

时间:2013-08-01 10:09:42

标签: c# linq entity-framework lambda entity-framework-5

我想创建像这样的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 does not support non-scalar variable

我按照建议here将lambda更改为user => user.AddressId == addressId。如何从已知的导航属性AddressId获取PropertyInfo FK Address

1 个答案:

答案 0 :(得分: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 }
   );
}