Java创建二叉树,并实现递归、非递归以及层序遍历

二叉树的基本知识

树的定义:

树是一种数据结构,它是由n(n>=1)个有限结点组成一个具有层次关系的集合。
当n=0时,称为空树

树的概述图

树具有的特点有:

1、每个结点有零个或多个子结点
2、每一个非根结点有且只有一个父节点
3、除了根结点外,每个子结点可以分为多个不相交的子树。
4、一颗n个结点的树有n-1条边(从下往上看,根节点不提供边)

树的一些基本术语:

1、结点的度:结点拥有的子树的数目
2、树的度:树中结点的最大的度
3、叶结点:度为0的结点
4、父结点:有子树的结点是其子树根节点的父节点
5、子节点,F是C的父结点,C是F的子结点
6、兄弟结点,具有同一父结点的各结点彼此是兄弟结点
7、祖先结点,沿树到某一结点路径上所有结点都是这个结点的祖先结点
8、子孙结点
9、结点的层次:根结点的层次为1,其余结点的层次等于其父结点的层次加1
10、树的深度:树中结点的最大层次

二叉树的定义:

二叉树是每个结点最多有两个子树的树结构。它有五种基本形态:二叉树可以是空集;根和空的左子右子树;根和左子树或右子树;根和左、右子树。

满二叉树和完全二叉树:

1、满二叉树(完美二叉树)

2、完全二叉树

定义:对n个结点的树,从上到下、从左到右顺序进行编号,任一结点编号与二叉树中相同编号结点位置相同。

一棵满二叉树是一棵完全二叉树,但完全二叉树未必是满二叉树。

二叉树的性质:

性质1:二叉树第i层上的最大结点数为2^{i-1}(i>=1)

性质2:深度为k的二叉树有最大结点总数2^k-1(k>=1)

性质3:在任意一棵二叉树中,若叶结点的个数为n0,度为2的结点数为n2,则n0=n2+1

性质证明03:19开始

创建一颗如下图的树,并进行递归、非递归(前序、中序、后序)以及层序遍历

代码实现如下:

import java.util.ArrayList;
import java.util.List;
import java.util.Queue;
import java.util.Stack;
import java.util.concurrent.LinkedBlockingQueue;

public class MyTree {
    public TreeNode root; //根结点
    public List<TreeNode> datas; //结点集合

    public MyTree() {
        this.root = new TreeNode(); //初始化
    }

    //创建树
    public void CreatTree(Object[] data) {
        datas = new ArrayList<TreeNode>();
        //将数组中的待添加的元素全部构建成树结点,放入datas集合中
        for (Object o : data) {
            datas.add(new TreeNode(o));
        }
        root = datas.get(0); //根结点更新为集合第一个元素
        for (int i = 0; i <= data.length / 2 - 1; i++) { //i限制不超过这个值,也就是叶结点没有子树
            if (datas.get(i) != null) { //为null时,也就是空节点了,没有子树
                datas.get(i).setLeft(datas.get(2 * i + 1));
                datas.get(i).setRight(datas.get(2 * i + 2));
            }
        }
    }

    //递归前序遍历
    public void PreOrderTraversal(TreeNode root) {
        if (root != null) {
            System.out.println(root.getData());
            PreOrderTraversal(root.getLeft());
            PreOrderTraversal(root.getRight());
        }
    }

    //递归中序遍历
    public void InOrderTraversal(TreeNode root) {
        if (root != null) {
            InOrderTraversal(root.getLeft());
            System.out.println(root.getData());
            InOrderTraversal(root.getRight());
        }
    }

    //递归后序遍历
    public void PostOrderTraversal(TreeNode root) {
        if (root != null) {
            PostOrderTraversal(root.getLeft());
            PostOrderTraversal(root.getRight());
            System.out.println(root.getData());
        }
    }

    //非递归中序遍历
    public void SinOrderTraversal(TreeNode root) {
        Stack<TreeNode> stack = new Stack<TreeNode>(); //递归是靠堆栈实现的,非递归直接用堆栈
        while (null != root || !stack.isEmpty()) { //遍历
            //左子树入栈
            while (root != null) {
                stack.push(root);
                root = root.getLeft();
            }
            //左子树入栈完毕
            if (!stack.isEmpty()) {
                TreeNode pop = stack.pop();
                System.out.println(pop.getData()); //按照走的路径,第2次碰见出栈就打印,叶结点第1次、第2次、第3次是同时碰见的
                root = pop.getRight();
            }
        }
    }

    //非递归前序遍历
    public void SpreOrderTraversal(TreeNode root) {
        Stack<TreeNode> stack = new Stack<TreeNode>();
        while (null != root || !stack.isEmpty()) { //遍历
            //左子树入栈
            while (root != null) {
                stack.push(root);
                System.out.println(root.getData()); //按照走的路径,第1次碰见入栈就打印
                root = root.getLeft();
            }
            //左子树入栈完毕
            if (!stack.isEmpty()) {
                TreeNode pop = stack.pop();
                root = pop.getRight();
            }
        }
    }

    //非递归后序遍历
    public void SpostOrderTraversal(TreeNode root) {
        Stack<TreeNode> stack = new Stack<TreeNode>();
        while (null != root || !stack.isEmpty()) { //遍历
            //左子树入栈
            while (root != null) {
                stack.push(root);
                root = root.getLeft();
            }
            //左子树入栈完毕
            boolean flag = false; //状态记录,来记录是否遍历了右子树(如果有),false为没有遍历,true为遍历了
            TreeNode pre = null; //记录上一个输出结点
            while (!stack.isEmpty() && flag == false) {
                if (stack.peek().getRight() == pre) { //如果栈的顶层结点没有(null)右子树或者是右子树上一次(pre)输出了
                    TreeNode pop = stack.pop(); //出栈
                    System.out.println(pop.getData());
                    pre = pop; //更新输出记录
                } else {
                    //如果有右子树并且没有输出,准备遍历右子树,并更新flag状态,跳出循环
                    root = stack.peek().getRight();
                    flag = true;
                }
            }
        }
    }

    //层序遍历
    /*
    根结点入队
    循环:结点出队,该结点左右左右儿子入队
     */
    public void LevelOrderTraversal(TreeNode root) {
        Queue<TreeNode> queue = new LinkedBlockingQueue<>();
        queue.add(root);
        while (root != null) {
            TreeNode node = queue.remove();
                System.out.println(node.getData());
            if (node.getLeft() != null) {
                queue.add(node.getLeft());
            }
            if (node.getRight() != null) {
                queue.add(node.getRight());
            }
            root = queue.peek();
        }
    }
}

//main测试
public static void main(String[] args) {
        MyTree tree = new MyTree();
        Object[] arr = {"A", "B", "C", "D", "F", "G", "I", null, null, "E", null, null, "H"};
        tree.CreatTree(arr);
        System.out.println("--递归前序--");
        tree.PreOrderTraversal(tree.root); //A\B\D\null\null\F\E\null\C\G\null\H\I
        System.out.println("--递归中序--");
        tree.InOrderTraversal(tree.root); //null\D\null\B\E\F\null\A\null\G\H\C\I
        System.out.println("--递归后序--");
        tree.PostOrderTraversal(tree.root); //null\null\D\E\null\F\B\null\H\G\I\C\A
        System.out.println("--非递归前序--");
        tree.SpreOrderTraversal(tree.root); //A\B\D\null\null\F\E\null\C\G\null\H\I
        System.out.println("--非递归中序--");
        tree.SinOrderTraversal(tree.root); //null\D\null\B\E\F\null\A\null\G\H\C\I
        System.out.println("--非递归后续--");
        tree.SpostOrderTraversal(tree.root); //null\null\D\E\null\F\B\null\H\G\I\C\A
        System.out.println("--层序遍历--");
        tree.LevelOrderTraversal(tree.root); //A\B\C\D\F\G\I\null\null\E\null\null\H
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,820评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,648评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,324评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,714评论 1 297
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,724评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,328评论 1 310
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,897评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,804评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,345评论 1 318
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,431评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,561评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,238评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,928评论 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,417评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,528评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,983评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,573评论 2 359