扩展表达式<func <t,bool =“”>&gt;

时间:2018-04-28 12:36:04

标签: c# expression-trees

自从我深入研究表达树以来,已经有很长一段时间了,我正在努力解决以下问题。

基本上我想采取以下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?

    }

}

1 个答案:

答案 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是适配器的实例。