数据结构—二叉树深度优先遍历

二叉树是一种常见的数据结构,理解二叉树对于理解AVL树、红黑树都有重要意义,索性再重新梳理一下思路,加深印象。本文重点介绍二叉查找树。

1.二叉树与二叉查找树

二叉树(binary tree)是树的一种特殊形式,这种树的每个节点做多只有两个孩子节点,二叉树结构如图:
数据结构—二叉树深度优先遍历_第1张图片
二叉树的树形结构使它很适合扮演索引的角色,因此出现了一种特殊的二叉树—二叉查找树(binary search tree),二叉查找树在二叉树的基础上,又增加了几个条件:
1)如果左子树不为空,则左子树上所有的节点值均小于根节点值;
2)如果右子树不为空,则右子树上所有节点的值均大于根节点的值;
3)左右子树也都是二叉查找树

下图就是一个典型的二叉查找树:
数据结构—二叉树深度优先遍历_第2张图片

2.二叉树的深度优先遍历

2.1前序遍历

二叉树的前序遍历,输出顺序是根节点、左子树、右子树。示意图如图所示:
1)首先输出根节点6;
2)由于根节点6存在左子节点,输出左子节点3;
3)由于节点3也存在子节点,输出左子节点2;
4)节点2没有左子节点,也没右子节点,回到节点3,输出节点3右子节点5;
5)节点5既没左子节点,也没右子节点,回到节点6,输出节点6的右子节点8;
6)节点8没有左子节点,有右子节点,因此输出节点8的右子节点9;
到此为止,所有节点输出完毕。
数据结构—二叉树深度优先遍历_第3张图片

2.2中序遍历

二叉树的前序遍历,输出顺序是左子树、根节点、右子树。示意图如图所示:
1)首先访问左子节点,如果左子节点还有左子节点,则继续深入访问,一直找到不再有左子节点的节点,输出节点2;
2)输出节点2的父节点节点3;
3)输出节点2的右节点节点5;
4)以节点2为根的左子树已输出完毕,这时输出整个二叉树的根节点节点6;
5)由于节点8没左子节点,直接输出节点8;
6)最后输出节点8的右子树节点9;
到此为止,所有节点输出完毕。
数据结构—二叉树深度优先遍历_第4张图片

2.3后序遍历

二叉树的前序遍历,输出顺序是左子树、右子树、根节点。示意图如图所示:
1)首先访问左子节点,如果左子节点还有左子节点,则继续深入访问,一直找到不再有左子节点的节点,输出节点2;
2)输出节点2的右节点节点5;
3)输出节点2的父节点节点3;
4)以节点2为根的左子树已输出完毕,开始查找根节点的右子树,节点8没有左子节点,但有右子节点,输出右子节点9;
5)输出右子节点的父节点节点8;
6)最后输出节根节点节点6;
到此为止,所有节点输出完毕。
数据结构—二叉树深度优先遍历_第5张图片

3.配套代码

三种遍历方式,用递归实现会非常简单。

/**
 * 二叉查找树
 */
public class BinarySearchTree {

    //根节点
    TreeNode root;

    /**
     * 构建二叉查找树
     * @param node
     * @param data
     */
    public void createTree(TreeNode node, int data){
        if(root == null){
            root = new TreeNode(data);
        }else{
            if(data < node.data){
                if(node.left == null){
                    node.left = new TreeNode(data);
                }else{
                    createTree(node.left,data);
                }
            }else{
                if(node.right == null){
                    node.right = new TreeNode(data);
                }else{
                    createTree(node.right,data);
                }
            }
        }
    }

    /**
     *二叉树前序遍历
     * @param node
     */
    public void preOrderTraversal(TreeNode node){
        if(node == null){
            return ;
        }
        System.out.println(node.data);
        preOrderTraversal(node.left);
        preOrderTraversal(node.right);
    }
    /**
     *二叉树中序遍历
     * @param node
     */
    public void middleOrderTraversal(TreeNode node){
        if(node == null){
            return ;
        }
        middleOrderTraversal(node.left);
        System.out.println(node.data);
        middleOrderTraversal(node.right);
    }
    /**
     *二叉树后序遍历
     * @param node
     */
    public void postOrderTraversal(TreeNode node){
        if(node == null){
            return ;
        }
        postOrderTraversal(node.left);
        postOrderTraversal(node.right);
        System.out.println(node.data);
    }
}
/**
 * 二叉树节点
 */
class TreeNode {
    //节点数据
    int data;
    //左子树
    TreeNode1 left;
    //右子树
    TreeNode1 right;

    public TreeNode(int data) {
        this.data = data;
    }
}

测试类:

public  static void main(String[] args){

        int[] arr = {10,2,3,1,11};
        BinarySearchTree tree = new BinarySearchTree();
        for(int data : arr){
            tree.createTree(tree.root,data);
        }
        //前序遍历
        System.out.println("********前序遍历********");
        tree.preOrderTraversal(tree.root);
        //中序遍历
        System.out.println("********中序遍历********");
        tree.middleOrderTraversal(tree.root);
        //后序遍历
        System.out.println("********后序遍历********");
        tree.postOrderTraversal(tree.root);
 }

你可能感兴趣的:(数据结构,数据结构)