我有一个BinaryTree类和一个BinaryTreeNode用于保存节点,我已经创建了树并为它编写了pre-order、postorder和order方法。
但我不知道如何为它编写IEnumerator<T>(我只想为GetEnumerator()方法使用)。
问题是inOrder方法的返回类型是void我想让它IEnumerator<T>而不是MessageBox返回数据。
我该怎么做?

public void PreOrder(BinaryTreeNode<T> node)
{
    if (node != null)
    {
        MessageBox.Show(node.Value.ToString());
        PreOrder(node.Left);
        PreOrder(node.Right);
    }
}

public void PostOrder(BinaryTreeNode<T> node)
{
    if (node != null)
    {
        PostOrder(node.Left);
        PostOrder(node.Right);
        MessageBox.Show(node.Value.ToString());
    }
}

public void InOrder(BinaryTreeNode<T> node)
{
    if (node != null)
    {
        InOrder(node.Left);
        MessageBox.Show(node.Value.ToString());
        InOrder(node.Right);
    }
}

public void Clear()
{
    root = null;
    Count = 0;
}

public IEnumerator<T> GetEnumerator()
{
    InOrder(root);
    return null; // error in forerach loop
}

IEnumerator IEnumerable.GetEnumerator()
{
    return GetEnumerator();
}

我觉得这部分代码已经足够了。
我就是这样定义BinaryTree类的:
public class BinaryTree<T> : IEnumerable<T> where T : IComparable<T>

我就是这样定义的:
public class BinaryTreeNode<T> : IComparable<T> where T : IComparable<T>

最佳答案

这里有一种方法,使用yield

public IEnumerator<T> GetEnumerator()
{
    if (Left != null)
    {
        foreach(var v in Left)
        {
            yield return v;
        }
    }

    yield return Value;

    if (Right != null)
    {
        foreach (var v in Right)
        {
            yield return v;
        }
    }
}

下面是使用Linq的更简洁的方法:
public IEnumerator<T> GetEnumerator()
{
    var leftEnumerable = (IEnumerable<T>)Left ?? new T[0];
    var rightEnumerable = (IEnumerable<T>)Right ?? new T[0];

    return leftEnumerable.Concat(new T[] { Value })
                         .Concat(rightEnumerable)
                         .GetEnumerator();
}

编辑:由于您似乎对BinaryTreeBinaryTreeNode有单独的类,您可以将以上任何一个放在BinaryTreeNode中,并将以下内容放在BinaryTree中:
public IEnumerator<T> GetEnumerator()
{
    return Root.GetEnumerator();
}

10-08 01:24