Как отмечают Джон Скит и полковник Паник в своих ответах, использование yield return
в рекурсивных методах может вызвать проблемы с производительностью, если дерево очень глубокое.
Вот общий нерекурсивный метод расширения, который выполняет обход глубины последовательности деревьев:
public static IEnumerable<TSource> RecursiveSelect<TSource>(
this IEnumerable<TSource> source, Func<TSource, IEnumerable<TSource>> childSelector)
{
var stack = new Stack<IEnumerator<TSource>>();
var enumerator = source.GetEnumerator();
try
{
while (true)
{
if (enumerator.MoveNext())
{
TSource element = enumerator.Current;
yield return element;
stack.Push(enumerator);
enumerator = childSelector(element).GetEnumerator();
}
else if (stack.Count > 0)
{
enumerator.Dispose();
enumerator = stack.Pop();
}
else
{
yield break;
}
}
}
finally
{
enumerator.Dispose();
while (stack.Count > 0) // Clean up in case of an exception.
{
enumerator = stack.Pop();
enumerator.Dispose();
}
}
}
В отличие от решения Эрика Липперта , RecursiveSelect работает напрямую со счетчиками, поэтому ему не нужно вызывать Reverse (который буферизует всю последовательность в памяти).
Используя RecursiveSelect, оригинальный метод OP можно переписать просто так:
public static IEnumerable<Control> GetDeepControlsByType<T>(this Control control)
{
return control.Controls.RecursiveSelect(c => c.Controls).Where(c => c is T);
}