自从我深入研究表达树以来,已经有很长一段时间了,我正在努力解决以下问题。
基本上我想采取以下Expression<Func<T, TIdType>>
:
(a) => EF.Property<TIdType>(a, "TenantId")
并将其扩展为此Expression<Func<T, bool>
:
(a) => EF.Property<TIdType>(a, "TenantId").Equals(TenantId)
所以基本上我想采用原始表达式并添加.Equals(TenantId)
对于此问题的背景,一切都归功于我尝试解决EF Core 2.0中的问题,如此处所示:
https://github.com/aspnet/EntityFrameworkCore/issues/11456
以下希望更详细地展示我正在尝试的内容:
public class FooEntity
{
public Guid TenantId { get; set; }
}
[TestClass]
public class UnitTest1
{
[TestMethod]
public void TestMethod1()
{
var adaptor = new TenantFilterExpressionAdaptor<FooEntity, Guid>();
var tenantIdFilter = adaptor.Adapt((a) => EF.Property<Guid>(a, "TenantId"));
}
}
public class TenantFilterExpressionAdaptor<T, TIdType>
{
public TIdType TenantId { get; set; }
public Expression<Func<T, bool>> Adapt(Expression<Func<T, TIdType>> idExpression)
{
// Tack on .Equals(TenantId) to the expression!?
// assume I have to add nodes to the body?
// idExpression.Body?
}
}
答案 0 :(得分:4)
您必须构建新表达式,因为表达式是不可变的。在这种情况下,您可以为新的表达式重用现有表达式的主体和参数:
public class TenantFilterExpressionAdaptor<T, TIdType> {
public TIdType TenantId { get; set; }
public Expression<Func<T, bool>> Adapt(Expression<Func<T, TIdType>> idExpression) {
return Expression.Lambda<Func<T, bool>>(
Expression.Equal(idExpression.Body, Expression.Constant(TenantId)),
idExpression.Parameters);
}
}
你在评论中提到,常数对你来说是个问题。在这种情况下,您可以引用属性本身而不是其值:
return Expression.Lambda<Func<T, bool>>(
Expression.Equal(idExpression.Body, Expression.Property(Expression.Constant(this), nameof(TenantId))),
idExpression.Parameters);
这将表达如下:
x => x.TenantId == this.TenantId
其中this
是适配器的实例。