我想创建像这样的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
。
答案 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 }
);
}