EF Core 1.0:如何查询整个树

本文关键字:查询 何查询 Core EF | 更新日期: 2023-09-27 18:37:20

我正在使用EF 7.0.0.0-rc1-final。

我有一个树结构,其中包含从祖祖父到祖父到父到子的一对多关系:

public class GrandGrandParent
{
    public int ID { get; set; }
    public string Name { get; set; }
    public virtual List<GrandParent> GrandParents { get; set; }
    public GrandGrandParent()
    {
        this.GrandParents = new List<GrandParent>();
    }
}
public class GrandParent
{
    public int ID { get; set; }
    public string Name { get; set; }
    public virtual GrandGrandParent GrandGrandParent { get; set; }
    public virtual List<Parent> Parents { get; set; }
    public GrandParent()
    {
        this.Parents = new List<Parent>();
    }
}
public class Parent
{
    public int ID { get; set; }
    public string Name { get; set; }
    public virtual GrandParent GrandParent { get; set; }
    public virtual List<Child> Children { get; set; }
    public Parent()
    {
        this.Children = new List<Child>();
    }
}
public class Child
{
    public int ID { get; set; }
    public string Name { get; set; }
    public virtual Parent Parent { get; set; }
}

使用 EF Core 1.0 (EF 7),如何创建 LINQ 查询(或使用子查询),在给定某个祖父 ID 的情况下为我提供整个树?

我可以。包括()向上或向下一级,也许我对显而易见的视而不见?这给了我祖父母和祖父母名单:

var ggparent1 = from ggp in myDbContext.GrandGrandParent
                .Include(ggp => ggp.GrandParents)
                where ggp.ID == 2
                select ggp;

我想把整棵树,直到孩子名单。我必须求助于 foreach() 循环并手动构建树吗?

EF Core 1.0:如何查询整个树

我只会使用 Linq 形式:

private static void Test0(ApplicationDbContext myDbContext)
{
    var ggparent = myDbContext.GrandGrandParents
        .Include(ggp => ggp.GrandParents)
        .ThenInclude(gp => gp.Parents)
        .ThenInclude(p => p.Children)
        .FirstOrDefault(ggp => ggp.ID == 3);
    if (ggparent == null)
    {
        DebugPrint("GrandGrandParent not found");
        return;
    }
    DebugPrint("GrandGrandParent:");
    DebugPrint(ggparent);

    if (ggparent.GrandParents == null)
    {
        DebugPrint("GrandParents null");
        return;
    }
    foreach (var gparent in ggparent.GrandParents)
    {
        DebugPrint(gparent);
        if (gparent.Parents == null) continue;
        foreach (var parent in gparent.Parents)
        {
            DebugPrint(parent);
            if (parent.Children == null) continue;
            foreach (var child in parent.Children)
            {
                DebugPrint(child);
            }
        }
    }
    int changeCount = myDbContext.SaveChanges();
    DebugPrint(string.Format("ChangeCount={0}", changeCount));
}

然后,您不必自己执行子查询。但可以添加日志记录以查看 EF 实际创建的 SQL 查询。

如果您需要完整的树,则不需要select。可以使用 select 提取结果树的一部分,或从层次结构中构建新的(平面)对象。

例如,我平展只有两个级别(多对多)的层次结构以获取分组选择列表的项目:

var result =
    (from c in dbContext.EventTypes
    join j in dbContext.EventType2EventTypes on c.Id equals j.ChildEventTypeId
    join p in dbContext.EventTypes on j.ParentEventTypeId equals p.Id
    where p.EventTypeLevel == EventTypeLevel.First && c.EventTypeLevel == EventTypeLevel.Second
    orderby p.SortOrder, p.Id, c.SortOrder, c.Id
    select new SelectListItem
    {
        Text = c.NameDe,
        Value = c.Id.ToString(),
        Group = GetParentEventTypeSelectListGroup(p.NameDe)
    }).AsNoTracking();