【完全模拟递归】二叉树迭代遍历

2021-09-10  本文已影响0人  蓝笔头

生成二叉树的工具,参考:构建二叉树
打印二叉树的工具类,参考:打印二叉树
递归遍历二叉树,参考:二叉树递归遍历

迭代模拟递归

public class BinaryTreeStackPrinter {

    public static void iterationPrint(TreeNode root) {
        if (root == null) return;

        Stack<TreeNode> stack = new Stack<>();
        stack.push(root);
        iterationPrint(stack);
    }

    private static void iterationPrint(Stack<TreeNode> stack) {
        // newStackFrame == true 表示当前有新节点入栈
        boolean newStackFrame = true;

        // lastVisisted 记录最近一次从栈中弹出的节点,避免重复入栈
        TreeNode lastVisited = new TreeNode(-1);

        while (!stack.isEmpty()) {
            TreeNode root = stack.peek();
            if (newStackFrame) {
                System.out.println(root.val + " 【入栈】                   | " + stack);
            }

            // 因为先访问左子树,所以如果左子树 or 右子树被访问过,说明左子树已经被访问过了
            boolean leftNodeVisited = (root.left == lastVisited || root.right == lastVisited);

            // 左子树不为空,且没有被访问过,才压栈
            if (root.left != null && !leftNodeVisited) {
                stack.push(root.left);
                newStackFrame = true;
                continue; // 通过 continue 模拟递归调用
            }

            boolean rightNodeVisited = (root.right == lastVisited);

            // 右子树不为空,且没有被访问过,才压栈
            if (root.right != null && !rightNodeVisited) {
                stack.push(root.right);
                newStackFrame = true;
                continue; // 通过 continue 模拟递归调用
            }

            root = stack.pop();
            System.out.println("----------" + root.val + "] 【出栈】        | " + stack);

            // 弹出栈中元素,会回退到上一个压栈元素,因此不会创建一个新的栈帧
            newStackFrame = false;

            // 记录最后一个弹出的元素,避免刚出栈又重新入栈
            lastVisited = root;
        }
    }
}

三种遍历方式

public class BinaryTreeStackPrinter {

    public static void iterationPrint(TreeNode root) {
        if (root == null) return;

        Stack<TreeNode> stack = new Stack<>();
        stack.push(root);
        iterationPrint(stack);
    }

    private static void iterationPrint(Stack<TreeNode> stack) {
        // newStackFrame == true 表示当前有新节点入栈
        boolean newStackFrame = true;

        // lastVisisted 记录最近一次从栈中弹出的节点,避免重复入栈
        TreeNode lastVisited = new TreeNode(-1);

        while (!stack.isEmpty()) {
            TreeNode root = stack.peek();
            if (newStackFrame) {
                System.out.println(root.val + " 【入栈】                   | " + stack);

//                System.out.println("【前序遍历】" + root.val);
            }

            // 因为先访问左子树,所以如果左子树 or 右子树被访问过,说明左子树已经被访问过了
            boolean leftNodeVisited = (root.left == lastVisited || root.right == lastVisited);

            // 左子树不为空,且没有被访问过,才压栈
            if (root.left != null && !leftNodeVisited) {
                stack.push(root.left);
                newStackFrame = true;
                continue; // 通过 continue 模拟递归调用
            }

            boolean rightNodeVisited = (root.right == lastVisited);
            if (!rightNodeVisited) {
//                System.out.println("【中序遍历】" + root.val);
            }

            // 右子树不为空,且没有被访问过,才压栈
            if (root.right != null && !rightNodeVisited) {
                stack.push(root.right);
                newStackFrame = true;
                continue; // 通过 continue 模拟递归调用
            }

            root = stack.pop();
            System.out.println("----------" + root.val + "] 【出栈】        | " + stack);

//            System.out.println("【后序遍历】" + root.val);

            // 弹出栈中元素,会回退到上一个压栈元素,因此不会创建一个新的栈帧
            newStackFrame = false;

            // 记录最后一个弹出的元素,避免刚出栈又重新入栈
            lastVisited = root;
        }
    }
}

测试类:

public class BinaryTreeDemo {
    public static void main(String[] args) {
        TreeNode root = new BinaryTreeBuilder(3, 0.6, 0.6).build();
        BinaryTreePrinter.print(root);

        String prefixAndSuffix = String.join("", Collections.nCopies(20, "*"));
        System.out.println(prefixAndSuffix + "【递归遍历】" + prefixAndSuffix);
        BinaryTreeStackPrinter.recursivePrint(root);

        System.out.println();

        System.out.println(prefixAndSuffix + "【迭代遍历】" + prefixAndSuffix);
        BinaryTreeStackPrinter.iterationPrint(root);
    }
}
maxLevel: 3
level[1] 节点宽度[ 16]===|          1       
                     ===|      ┌───┴───┐   
level[2] 节点宽度[  8]===|      2       5   
                     ===|    ┌─┴─┐   ┌─┘   
level[3] 节点宽度[  4]===|    3   4   6     
********************【递归遍历】********************
1 【入栈】                   | [1]
【前序遍历】1
2 【入栈】                   | [1, 2]
【前序遍历】2
3 【入栈】                   | [1, 2, 3]
【前序遍历】3
----------3【出栈】          | [1, 2]
4 【入栈】                   | [1, 2, 4]
【前序遍历】4
----------4【出栈】          | [1, 2]
----------2【出栈】          | [1]
5 【入栈】                   | [1, 5]
【前序遍历】5
6 【入栈】                   | [1, 5, 6]
【前序遍历】6
----------6【出栈】          | [1, 5]
----------5【出栈】          | [1]
----------1【出栈】          | []

********************【迭代遍历】********************
1 【入栈】                   | [1]
【前序遍历】1
2 【入栈】                   | [1, 2]
【前序遍历】2
3 【入栈】                   | [1, 2, 3]
【前序遍历】3
----------3] 【出栈】        | [1, 2]
4 【入栈】                   | [1, 2, 4]
【前序遍历】4
----------4] 【出栈】        | [1, 2]
----------2] 【出栈】        | [1]
5 【入栈】                   | [1, 5]
【前序遍历】5
6 【入栈】                   | [1, 5, 6]
【前序遍历】6
----------6] 【出栈】        | [1, 5]
----------5] 【出栈】        | [1]
----------1] 【出栈】        | []
maxLevel: 3
level[1] 节点宽度[ 16]===|          1       
                     ===|      ┌───┴───┐   
level[2] 节点宽度[  8]===|      2       4   
                     ===|    ┌─┘       └─┐ 
level[3] 节点宽度[  4]===|    3           5 
********************【递归遍历】********************
1 【入栈】                   | [1]
2 【入栈】                   | [1, 2]
3 【入栈】                   | [1, 2, 3]
【中序遍历】3
----------3【出栈】          | [1, 2]
【中序遍历】2
----------2【出栈】          | [1]
【中序遍历】1
4 【入栈】                   | [1, 4]
【中序遍历】4
5 【入栈】                   | [1, 4, 5]
【中序遍历】5
----------5【出栈】          | [1, 4]
----------4【出栈】          | [1]
----------1【出栈】          | []

********************【迭代遍历】********************
1 【入栈】                   | [1]
2 【入栈】                   | [1, 2]
3 【入栈】                   | [1, 2, 3]
【中序遍历】3
----------3] 【出栈】        | [1, 2]
【中序遍历】2
----------2] 【出栈】        | [1]
【中序遍历】1
4 【入栈】                   | [1, 4]
【中序遍历】4
5 【入栈】                   | [1, 4, 5]
【中序遍历】5
----------5] 【出栈】        | [1, 4]
----------4] 【出栈】        | [1]
----------1] 【出栈】        | []
maxLevel: 3
level[1] 节点宽度[ 16]===|          1       
                     ===|      ┌───┴───┐   
level[2] 节点宽度[  8]===|      2       5   
                     ===|    ┌─┴─┐   ┌─┘   
level[3] 节点宽度[  4]===|    3   4   6     
********************【递归遍历】********************
1 【入栈】                   | [1]
2 【入栈】                   | [1, 2]
3 【入栈】                   | [1, 2, 3]
----------3【出栈】          | [1, 2]
【后序遍历】3
4 【入栈】                   | [1, 2, 4]
----------4【出栈】          | [1, 2]
【后序遍历】4
----------2【出栈】          | [1]
【后序遍历】2
5 【入栈】                   | [1, 5]
6 【入栈】                   | [1, 5, 6]
----------6【出栈】          | [1, 5]
【后序遍历】6
----------5【出栈】          | [1]
【后序遍历】5
----------1【出栈】          | []
【后序遍历】1

********************【迭代遍历】********************
1 【入栈】                   | [1]
2 【入栈】                   | [1, 2]
3 【入栈】                   | [1, 2, 3]
----------3] 【出栈】        | [1, 2]
【后序遍历】3
4 【入栈】                   | [1, 2, 4]
----------4] 【出栈】        | [1, 2]
【后序遍历】4
----------2] 【出栈】        | [1]
【后序遍历】2
5 【入栈】                   | [1, 5]
6 【入栈】                   | [1, 5, 6]
----------6] 【出栈】        | [1, 5]
【后序遍历】6
----------5] 【出栈】        | [1]
【后序遍历】5
----------1] 【出栈】        | []
【后序遍历】1

参考

上一篇下一篇

猜你喜欢

热点阅读