在自引用(父子)层次结构树中查找所有后代

时间:2016-05-25 15:20:09

标签: c# entity-framework

这类似于问题(Finding parents in a tree hierarchy for a given child LINQ (lambda expression))。但是,我不需要找到所有的祖先,而是找到所有的后代。

我正在修改Yacoub的方法,但只设法让所有后代都在一个分支中。

    private IEnumerable<UserRole> FindAllChildrenRecursively(List<UserRole> allRoles, UserRole role)
{
    var child = allRoles.FirstOrDefault(x => x.ParentId == role.Id);

    if (child == null)
        return Enumerable.Empty<UserRole>();

    return new[] { child }.Concat(FindAllChildrenRecursively(allRoles, child));
}

1 个答案:

答案 0 :(得分:4)

  

我正在修改Yacoub的方法,但只设法让所有后代都在一个分支中

这是因为这一行:

var child = allRoles.FirstOrDefault(x => x.ParentId == role.Id);

虽然找到一个单个父母可能是合适的,但它不适合找到多个孩子。

但是,您不需要递归迭代器和allRoles列表上的多次迭代。您可以使用ToLookup扩展方法创建快速查找结构,然后执行迭代DFS,如下所示:

private static IEnumerable<UserRole> FindAllChildren(List<UserRole> allRoles, UserRole role)
{
    var childrenByParentId = allRoles.ToLookup(r => r.ParentId);
    var stack = new Stack<IEnumerator<UserRole>>();
    var e = childrenByParentId[role != null ? role.Id : (int?)null].GetEnumerator();
    try
    {
        while (true)
        {
            while (e.MoveNext())
            {
                yield return e.Current;
                stack.Push(e);
                e = childrenByParentId[e.Current.Id].GetEnumerator();
            }
            if (stack.Count == 0) break;
            e.Dispose();
            e = stack.Pop();
        }
    }
    finally
    {
        e.Dispose();
        while (stack.Count > 0) stack.Pop().Dispose();
    }
}

更好的方法是(按照DRY原则)使用How to flatten tree via LINQ?中的通用树辅助方法:

public static class TreeUtils
{
    public static IEnumerable<T> Expand<T>(
            this IEnumerable<T> source, Func<T, IEnumerable<T>> elementSelector)
    {
        var stack = new Stack<IEnumerator<T>>();
        var e = source.GetEnumerator();
        try
        {
            while (true)
            {
                while (e.MoveNext())
                {
                    var item = e.Current;
                    yield return item;
                    var elements = elementSelector(item);
                    if (elements == null) continue;
                    stack.Push(e);
                    e = elements.GetEnumerator();
                }
                if (stack.Count == 0) break;
                e.Dispose();
                e = stack.Pop();
            }
        }
        finally
        {
            e.Dispose();
            while (stack.Count != 0) stack.Pop().Dispose();
        }
    }
}
像这样:

private static IEnumerable<UserRole> FindAllChildren(List<UserRole> allRoles, UserRole role)
{
    var childrenByParentId = allRoles.ToLookup(r => r.ParentId);
    return childrenByParentId[role != null ? role.Id : (int?)null].Expand(r => childrenByParentId[r.Id]);
}