c# - 是否可以将复杂的 EF Core ".Include"调用重写为 "Join"调用?

标签 c# linq asp.net-core entity-framework-core

我有两个名为 Service 和 ServiceBranch 的 EF Core 实体模型:

public class Service
{
    public Guid ID { get; set; }
    public Guid? ParentServiceID { get; set; } // foreign key to the same table
    public string Name { get; set; }
    public DateTime? DateDeleted { get; set; }

    public virtual ICollection<Service> InverseParent { get; set; } // navigation property
    public virtual ICollection<ServiceBranch> ServiceBranches { get; set; } // navigation property
}

public class ServiceBranch
{
    public Guid ID { get; set; }
    public Guid ServiceID { get; set; } // foreign key
    public string Name { get; set; }
    public DateTime? DateDeleted { get; set; }

    public virtual Service Service { get; set; } // navigation property
}

为简单起见,假设父服务只能有一个级别的子服务。许多服务分支都引用了父子服务。

我想更新特定服务 ID 的 DateDeleted 字段并更新引用它的所有其他行中的 DateDeleted(如果该字段尚无值)。

目前我调用所有必需的实体如下所示:

var serviceEntity = _context.Set<Service>().Where(x => x.ID == neededID && x.DateDeleted == null)
                .Include(x => x.ServiceBranches)
                .Include(x => x.InverseParent)
                    .ThenInclude(InverseParent => InverseParent.ServiceBranches)
                .FirstOrDefault();

我对主要实体使用 Where 子句,但遗憾的是我不能在 .Includes 中使用 Where,因为我们还没有升级到 EF Core 5.0(它支持在 .Includes 中进行过滤)。

因此我从数据库中获取了太多不需要的行(其中 DateDeleted 已经有一个值),然后通过将该字段与 null 进行比较来在 foreach 循环中过滤它们:

if (myentity.DateDeleted == null) myentity.DateDeleted = DateTime.Now;

我想重写我的查询,以便使用“join”而不是“.Include”,如下所示:

var serviceEntity = from service in _context.Set<Service>()
                where service.ID == neededID && service.DateDeleted == null
                join branches in _context.Set<ServiceBranch>() on service.ID equals branches.ServiceID
                where branches.DateDeleted == null // filter which I cannot use with .Include-s
                let // whatever, can't make it work
                select // etc

我不知道如何重写它。或者甚至可以通过联接来实现。

最佳答案

在这种特殊情况下,您不需要连接,而是包含所需的 Service 实体及其所有子实体的平面列表。一般来说,这需要递归查询(LINQ/EF Core 不支持),但对于单层来说,这是一个简单的过滤器问题,比如

service.ID == neededID || service.ParentServiceID == neededID

然后您可以应用附加的 DateDeleted 条件。最后,要使用 DateDeleted 过滤器获取相关的 ServiceBranch 实体,而不是 Include,只需使用投影(Select)。

并且在处理返回数据时,使用投影实体并忽略它们的导航属性。例如

var itemsToUpdate = _context.Set<Service>()
    .Where(service => (service.ID == neededID || service.ParentServiceID == neededID)
        && service.DateDeleted == null)
    .Select(service => new
    {
        Service = service,
        ServiceBrances = service.ServiceBranches
            .Where(branch => branch.DateDeleted == null),
    });

foreach (var item in itemsToUpdate)
{
    item.Service.DateDeleted = DateTime.Now;
    foreach (var branch in item.ServiceBrances)
        branch.DateDeleted =  DateTime.Now;
}

关于c# - 是否可以将复杂的 EF Core ".Include"调用重写为 "Join"调用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65747968/

相关文章:

C# LINQ 组合数学 : All Combinations of a Set without the Empty Set

c# - 强制选项卡到扩展的 .net 控件中的下一个控件

c# - 如何从对象中的对象抽象值?

html - 通过 Linq to XML 查询时如何处理任意命名空间?

c# - ASP.NET 5 跨平台,老式 DLL 将何去何从?

c# - Azure AD,无法针对邀请设置附加数据

asp.net-identity - 从我已经派生自 For RoleManager<TRole> 的东西派生的异常

c# - WPF DataGrid - 如何在验证错误后取消编辑行?

c# - .NET 引用源中的缩写词 EE 是什么意思?

c# - 自定义 header 中带有 ID 的 ASP.NET MVC session