C#Linq表达式无法翻译

时间:2020-03-03 10:14:46

标签: c# entity-framework linq entity-framework-core .net-core-3.0

我正在尝试执行linq查询,以获取具有某些特定技能的所有雇员。 search.skills是具有某些技能的字符串列表,我想检索所有具有所有这些技能(和条件)的用户。 在我关于员工的where子句中,exp.Skills是ICollection,expSkill.SkillName是技能名称

                .Where(
                emp => search.Skills.All(
                    searchSkill => emp.Experiences.Select(exp => exp.Skills).SelectMany(x => x).Select(expSkill => expSkill.SkillName).Contains(searchSkill)
                ))
            .ToListAsync();

尝试执行该错误时出现以下错误。我正在使用EntityFramework Core 3

The LINQ expression 'DbSet<Employee>
.Where(e => __search_Skills_0
    .All(searchSkill => DbSet<Experience>
        .Where(e0 => EF.Property<Nullable<Guid>>(e, "Id") != null && EF.Property<Nullable<Guid>>(e, "Id") == EF.Property<Nullable<Guid>>(e0, "EmployeeId"))
        .SelectMany(
            source: e0 => DbSet<ExperienceSkill>
                .Where(e1 => EF.Property<Nullable<Guid>>(e0, "EmployeeId") != null && new AnonymousObject(new object[]
                { 
                    (object)EF.Property<Nullable<Guid>>(e0, "EmployeeId"), 
                    (object)EF.Property<string>(e0, "ProjectCode") 
                }) == new AnonymousObject(new object[]
                { 
                    (object)EF.Property<Nullable<Guid>>(e1, "ExperienceEmployeeId"), 
                    (object)EF.Property<string>(e1, "ExperienceProjectCode") 
                })), 
            collectionSelector: (e0, c) => new TransparentIdentifier<Experience, ExperienceSkill>(
                Outer = e0, 
                Inner = c
            ))
        .Select(ti => ti.Inner.SkillName)
        .Contains(searchSkill)))' could not be translated. Either rewrite the query in a form that can be translated, or switch to client evaluation explicitly by inserting a call to either AsEnumerable(), AsAsyncEnumerable(), ToList(), or ToListAsync(). See https://go.microsoft.com/fwlink/?linkid=2101038 for more information.

谁能告诉我我在查询中做错了什么?谢谢

1 个答案:

答案 0 :(得分:3)

除诸如Contains之类的内存集合外,EF Core当前无法转换除简单search.Skills以外的LINQ运算符。

因此,您需要找到memoryCollection.All运算符的替代方法。在这种情况下,我使用的是“计数匹配”方法,该方法可以找到所有匹配的记录并比较不同值的计数。

例如替换

emp => search.Skills.All(
    searchSkill => emp.Experiences.Select(exp => exp.Skills).SelectMany(x => x).Select(expSkill => expSkill.SkillName).Contains(searchSkill)

使用

emp => emp.Experiences
    .SelectMany(exp => exp.Skills, (exp, skill) => skill.SkillName)
    .Where(skillName => search.Skills.Contains(skillName))
    .Distinct().Count() == search.Skills.Distinct().Count()