【问题标题】:Access children(list) related property in Expression tree访问表达式树中的子项(列表)相关属性
【发布时间】:2017-08-25 18:48:47
【问题描述】:

我为我的实体Master 创建了一个存储库。在存储库中,我有一个 Get 方法来使用 Entity Core 通过 Id 获取我的实体。

方法接收:

public TEntity Get(object id, params Expression<Func<TEntity, object>>[] includedRelatedEntities)
    {
        return GetById(IncludeEntities(DBContext.Set<TEntity>().AsQueryable(), includedRelatedEntities), id);
    }

然后,当我在我的代码中使用它时,我只需将我正在寻找的实体的 ID 和我需要包含在查询中的相关实体的表达式树传递给该方法 (Expression&lt;Func&lt;TEntity, object&gt;&gt;)

使用示例如下:

var master = MasterRepository.Get(1, x => x.BranchOffice.Location);

在这种情况下,我正在寻找 Id = 1 的 Master,我希望它包含 BranchOffice 相关实体和与 BranchOffice 相关的 Location

从一对多关系,它工作正常,但对于相关列表,我不知道如何使用表达式来解决它。

例如,如果我想包含与我的Master相关的Detail列表中名为DetailsProduct实体,我不知道如何在表达式树中表达。

var master = MasterRepository.Get(1, x => x.Details.Product);

Details 是一个列表,所以我无法访问上面示例中的产品。

如何在Expression&lt;Func&lt;TEntity, object&gt;&gt; 中表达这一点?

编辑:

我已经试过了:

var master = MasterRepository.Get(1, x => x.Details.Select(y=> y.Product));

但我得到以下异常:

属性表达式'x => {from Detail y in [x].Details select [y].Product}' 无效。表达式应该代表一个属性 访问:'t => t.MyProperty'。有关包括相关的更多信息 数据,请参阅 go.microsoft.com/fwlink/?LinkID=746393。'

【问题讨论】:

  • 请尝试使用Select 获取相关Product 以获取所有详细信息:var master = MasterRepository.Get(1, x =&gt; x.Details.Select(detail =&gt; detail.Product));
  • @GeorgeAlexandria 我已经尝试过了,但我遇到了一个异常:'属性表达式'x => {from Detail y in [x].Details select [y].Product}' 不是有效的。该表达式应表示属性访问:'t => t.MyProperty'。有关包含相关数据的更多信息,请参阅go.microsoft.com/fwlink/?LinkID=746393。'
  • 哦,我明白了,他们改变了Entity.Core中嵌套子项的检索...
  • 你能解释一下返回 IncludeEntities 是什么吗?它是否返回IQueryable

标签: c# lambda expression-trees


【解决方案1】:

我不知道您能否更改或替换 IncludeEntities 实现,所以也许回答对您没有帮助。好吧,x =&gt; x.Details.ProductEF.Core 中看起来像 DbContext.Set&lt;SomeType&gt;().Include(x =&gt; x.Details).ThenInclude(o =&gt; o.Product)

因此,如果您想包含多个级别,我建议您在运行时构建一个包含IncludeThenInclude 的查询。因此,此查询将根据输入表达式构建,如下所示 x =&gt; x.Details.Select(y =&gt; y.Product)。构建此查询的方法:

    /// <summary>
    /// Takes include looks like 'x => x.Collections.Select(o => o.List.Select(p => p.Date))'
    /// </summary>
    public static IQueryable<T> GetQueryWithIncludes<T>(IQueryable<T> query, Expression<Func<T, object>> arg)
    {
        // Tiny optimization
        ParameterInfo[] parameters;
        var includeInfo = typeof(EntityFrameworkQueryableExtensions).GetMethods().Where(info => info.Name == "Include" &&
            (parameters = info.GetParameters()).Length == 2 &&
            typeof(Expression).IsAssignableFrom(parameters[1].ParameterType)).Single();

        // Retrieve then include that take first param as 'IIncludableQueryable<TEntity, ICollection<TPreviousProperty>>'
        var thenIncludeInfo = typeof(EntityFrameworkQueryableExtensions).GetMethods().Where(info => info.Name == "ThenInclude").ToList()[1];
        // Retrieve then include that take first param as 'IIncludableQueryable<TEntity, IEnumerable<TPreviousProperty>>'
        var lastThenIncludeInfo = typeof(EntityFrameworkQueryableExtensions).GetMethods().Where(info => info.Name == "ThenInclude").ToList()[0];

        // Retrieve all selection from input expression
        var lambda = arg as LambdaExpression;
        var method = arg.Body as MethodCallExpression;
        var result = new List<Expression>();
        while (method != null)
        {
            result.Add(Expression.Lambda(method.Arguments[0], lambda.Parameters[0]));
            lambda = method.Arguments[1] as LambdaExpression;
            method = lambda.Body as MethodCallExpression;
        }
        result.Add(lambda);

        // Add Include and ThenInclude to IQueryable
        for (int i = 0; i < result.Count; ++i)
        {
            var lambdaExp = result[i] as LambdaExpression;
            query = i == 0
                ? includeInfo.MakeGenericMethod(lambdaExp.Parameters[0].Type, lambdaExp.ReturnType).Invoke(null, new object[] { query, lambdaExp }) as IQueryable<T>
                : i == result.Count - 1
                    ? lastThenIncludeInfo.MakeGenericMethod((result[0] as LambdaExpression).Parameters[0].Type, lambdaExp.Parameters[0].Type, lambdaExp.ReturnType).Invoke(null, new object[] { query, lambdaExp }) as IQueryable<T>
                    : thenIncludeInfo.MakeGenericMethod((result[0] as LambdaExpression).Parameters[0].Type, lambdaExp.Parameters[0].Type, lambdaExp.ReturnType).Invoke(null, new object[] { query, lambdaExp }) as IQueryable<T>;
        }
        return query;
    }

顺便说一下,方法接受一个表达式,但它可以稍微修改,所以它会接受一个表达式数组,或者你可以直接从循环中为所有表达式调用方法。

下面的代码只是用法。我写了一个树小类进行测试:

    public class Test
    {
        public int Id { get; set; }
        public DateTime TestDate { get; set; }
        public ICollection<Level> Levels { get; set; }
    }

    public class Level
    {
        public int Id { get; set; }
        public ICollection<LevelDetail> LevelDetails { get; set; }
    }

    public class LevelDetail
    {
        public int Id { get; set; }
        public DateTime LevelDate { get; set; }
    }

    ...
    // These results are the same and have the same expression trees
    var resultByInclude = context.Tests
        .Include(o => o.Levels)
        .ThenInclude(p => p.LevelDetails).ToList();

    var resultBySelect = GetQueryWithIncludes(context.Tests,
        o => o.Levels.Select(p => p.LevelDetails)).ToList();

希望对你有帮助。

【讨论】:

  • 谢谢。这就像一个魅力。无论如何,这是一种解决方法。也许 Entity Core 将来可能会支持这种 lambda 表达式。
  • @Andres 我怀疑。 EF6 Include 表达式语法在早期的 EFC 预发布版本中得到支持,然后在没有明显原因的情况下被删除。他们似乎坚持使用新的 Include / ThenInclude 模式,这在正常使用场景中可能看起来更好,但不能用像你的情况这样的表达式来表示。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2022-12-02
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2022-10-05
相关资源
最近更新 更多