二叉树的遍历和重建

二叉树是树的一种,由一个根节点和两棵互不相交左右子树的二叉树组成。
二叉树有很多性质,就不一一举例了~
先来说一下最简单的关于二叉树常用的3种遍历方式。层序遍历就略过了~

前序遍历:根节点,左节点,右节点。

结果就是:ABDGHCEIF


前序遍历
中序遍历:左节点,根节点,右节点。

结果就是:GDHBAEICF


中序遍历
后序遍历:左节点,右节点,根节点。

结果就是:GHDBIEFCA


后序遍历

概念就说到这里,开始敲代码~
首先,建立数节点。

public class TreeNode {

    private int index;      
 
    private String data;           //数据域 

    private TreeNode leftChild;    //左子树

    private TreeNode rightChild;   //右子树

    public int getIndex() {
        return index;
    }


    public void setIndex(int index) {
        this.index = index;
    }


    public String getData() {
        return data;
    }


    public void setData(String data) {
        this.data = data;
    }

     public TreeNodeString(String data) {
        this.data = data;
    }

    public TreeNode(int index, String data) {
        this.index = index;
        this.data = data;
        this.leftChild = null;
        this.rightChild = null;
    }
}

然后我们来构建一个二叉树:

/**
 * 构建二叉树
 *                  A
 *              B       C
 *          D       E      F
 *      G       H     I
 */
public void createBinaryTree() {
    //我们需要一个根节点
    root = new TreeNode(1, "A");   
    TreeNode treeNodeB = new TreeNode(2, "B");
    TreeNode treeNodeC = new TreeNode(3, "C");
    TreeNode treeNodeD = new TreeNode(4, "D");
    TreeNode treeNodeE = new TreeNode(5, "E");
    TreeNode treeNodeF = new TreeNode(6, "F");
    TreeNode treeNodeG = new TreeNode(7, "G");
    TreeNode treeNodeH = new TreeNode(8, "H");
    TreeNode treeNodeI = new TreeNode(9, "I");
    //分别对应节点的左右子树。
    root.leftChild = treeNodeB;
    root.rightChild = treeNodeC;
    treeNodeB.leftChild = treeNodeD;
    treeNodeD.leftChild = treeNodeG;
    treeNodeD.rightChild = treeNodeH;
    treeNodeC.leftChild = treeNodeE;
    treeNodeC.rightChild = treeNodeF;
    treeNodeE.rightChild = treeNodeI;

}

二叉树建立完成之后,
开始遍历,首先使用循环的方式进行前序遍历,循环需要借助栈来实现遍历。

/**
 *  前序遍历  非递归
 * @param treeNode
 */
public void nonRecOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }
    Stack<TreeNode> stack = new Stack<>();
    //首先把根节点放入栈中
    stack.push(treeNode);
    while (!stack.isEmpty()){
        //弹出根结点
        TreeNode node = stack.pop();
        System.out.println("data:" + node.data);
        //如果有右子结点,则压入节点
        if(node.rightChild!=null){
            //如果左节点有值,则右节点在栈底,最后才会输出
            stack.push(node.rightChild);
        }
        //如果有右左结点,则压入节点
        if(node.leftChild!=null){
            //下次循环把左子节点当根节点继续弹出
            stack.push(node.leftChild);
        }
    }
}

接着来介绍使用递归的方式进行遍历

/**
 *  前序遍历  递归
 * @param treeNode
 */
public void preOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }else{
        //首先把每个节点当成根节点输出
        System.out.println("data:" + treeNode.data);
        //如果左节点有值,则输出左节点
        preOrder(treeNode.leftChild);
        ///如果右节点有值,输出右节点。
        preOrder(treeNode.rightChild);
    }
}
/**
 *  中序遍历  递归
 * @param treeNode
 */
public void midOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }else{
        //递归左子树,直到左节点为null,return。
        midOrder(treeNode.leftChild);
        //输出节点。
        System.out.println("data:" + treeNode.data);
        //递归右子树
        midOrder(treeNode.rightChild);
    }
}
/**
 *  后序遍历  递归
 * @param treeNode
 */
public void postOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }else{
         //递归左子树,直到左节点为null,return。
        postOrder(treeNode.leftChild);
         //递归右子树,直到左节点为null,return。
        postOrder(treeNode.rightChild);
        //输出节点
        System.out.println("data:" + treeNode.data);
    }
}

二叉树的遍历理解了还是比较简单的,下面来看一下二叉树的重建。
已知 前序遍历结果 :ABDGHCEIF 中序遍历结果:GDHBAEICF,求二叉树。

根据前序遍历的特性,可以知道A是根节点,这种在中序中可以得出 A节点左边的就是左节点,右边的就是右节点。这个规律可以使用递归

整个二叉树的
左子树的

下面就来看看重建二叉树的递归代码,具体的在代码中都有注释

/**
 * 已知前序和中序
 * @param pre       前序集合
 * @param startPre  前序集合起始点
 * @param endPre    前序集合总长度
 * @param in        中序集合
 * @param startIn   中序集合起始点
 * @param endIn     中序集合总长度
 * @return
 */
public TreeNode resetBinaryTreeString(String[] pre ,int startPre,int  endPre,String[] in,int startIn,int endIn){
    //如果开始大于结束,就说明这个树已经结束了
    if(startPre > endPre || startIn > endIn) {
        return null;
    }
    //把前序的头节点放入树中当根节点
    TreeNode node = new TreeNode(pre[startPre]);
    //遍历中序的所有节点
    for (int i = startIn; i <= endIn; i++) {
        //找到根节点在中序中的位置
        if(pre[startPre] == in[i]){
           node.leftChild = resetBinaryTreeString(
                    pre,            //前序集合
                    startPre + 1,   //前序左子树起始点: startPre + 1
                    startPre + (i - startIn),//左子树长度:i是root节点在中序遍历的位置,
                                             // (i - startIn)是中序遍历中左子树的长度,
                                             // 所以左子树的总长度:前序的 startPre + 中序遍历中左子树的长度
                    in,             //中序集合
                    startIn,        //中序左子树的起始点:startIn
                    i - 1);         //中序左子树的总长度:i -1
            node.rightChild = resetBinaryTreeString(
                    pre,             //前序集合
                    startPre + (i - startIn) + 1,//前序右子树起始点: 前序左子树的长度+1 , {startPre + (i - startIn)} + 1
                    endPre,          //前序右子树总长度: 前序的总长度
                    in,              //中序集合
                    i + 1,           //中序右子树的起始点:i+1
                    endIn);          //中序右子树总长度: 中序的总长度
            break;
        }
    }
    return node;
}

这些是最近看二叉树的一些收获~简单的分享一下。

下面贴上全部代码:

public class BinaryTree {

private TreeNode root;

public BinaryTree() {
    root = new TreeNode(1, "A");
}

/**
 * 构建二叉树
 *                  A
 *              B       C
 *          D       E      F
 *      G       H     I
 */
public void createBinaryTree() {
    root = new TreeNode(1, "A");
    TreeNode treeNodeB = new TreeNode(2, "B");
    TreeNode treeNodeC = new TreeNode(3, "C");
    TreeNode treeNodeD = new TreeNode(4, "D");
    TreeNode treeNodeE = new TreeNode(5, "E");
    TreeNode treeNodeF = new TreeNode(6, "F");
    TreeNode treeNodeG = new TreeNode(7, "G");
    TreeNode treeNodeH = new TreeNode(8, "H");
    TreeNode treeNodeI = new TreeNode(9, "I");

    root.leftChild = treeNodeB;
    root.rightChild = treeNodeC;
    treeNodeB.leftChild = treeNodeD;
    treeNodeD.leftChild = treeNodeG;
    treeNodeD.rightChild = treeNodeH;
    treeNodeC.leftChild = treeNodeE;
    treeNodeC.rightChild = treeNodeF;
    treeNodeE.rightChild = treeNodeI;

}

/**
 *  前序遍历  递归
 * @param treeNode
 */
public void preOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }else{
        //首先输出根节点。
        System.out.println("data:" + treeNode.data);
        //不断的调用自身函数,把左节点输出。
        preOrder(treeNode.leftChild);
        //根节点和左节点输出完成,输出右节点。
        preOrder(treeNode.rightChild);
    }
}
/**
 *  中序遍历  递归
 * @param treeNode
 */
public void midOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }else{
        midOrder(treeNode.leftChild);
        System.out.println("data:" + treeNode.data);
        midOrder(treeNode.rightChild);
    }
}
/**
 *  后序遍历  递归
 * @param treeNode
 */
public void postOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }else{
        postOrder(treeNode.leftChild);
        postOrder(treeNode.rightChild);
        System.out.println("data:" + treeNode.data);
    }
}
/**
 *  前序遍历  非递归
 * @param treeNode
 */
public void nonRecOrder(TreeNode treeNode){
    if(treeNode == null){
        return;
    }
    Stack<TreeNode> stack = new Stack<>();
    stack.push(treeNode);
    while (!stack.isEmpty()){
        //弹出根结点
        TreeNode node = stack.pop();
        System.out.println("data:" + node.data);
        //如果有右子结点,则压入节点
        if(node.rightChild!=null){
            //如果左节点有值,则右节点在栈底,最后才会输出
            stack.push(node.rightChild);
        }
        //如果有右左结点,则压入节点
        if(node.leftChild!=null){
            //下次循环把左子节点当根节点继续弹出
            stack.push(node.leftChild);
        }
    }
}

/**
 * 已知前序和中序
 * @param pre       前序集合
 * @param startPre  前序集合起始点
 * @param endPre    前序集合总长度
 * @param in        中序集合
 * @param startIn   中序集合起始点
 * @param endIn     中序集合总长度
 * @return
 */
public TreeNode resetBinaryTreeString(String[] pre ,int startPre,int  endPre,String[] in,int startIn,int endIn){
    //如果开始大于结束,就说明这个树已经结束了
    if(startPre > endPre || startIn > endIn) {
        return null;
    }
    //把前序的头节点放入树中当根节点
    TreeNode node = new TreeNode(pre[startPre]);
    //遍历中序的所有节点
    for (int i = startIn; i <= endIn; i++) {
        //找到根节点在中序中的位置
        if(pre[startIn] == in[i]){
            node.leftChild = resetBinaryTreeString(
                    pre,            //前序集合
                    startPre + 1,   //前序左子树起始点: startPre + 1
                    startPre + (i - startIn),//左子树长度:i是root节点在中序遍历的位置,
                    // (i - startIn)是中序遍历中左子树的长度,
                    // 所以左子树的总长度:前序的 startPre + 中序遍历中左子树的长度
                    in,             //中序集合
                    startIn,        //中序左子树的起始点:startIn
                    i - 1);         //中序左子树的总长度:i -1
            node.rightChild = resetBinaryTreeString(
                    pre,             //前序集合
                    startPre + (i - startIn) + 1,//前序右子树起始点: 前序左子树的长度+1 , {startPre + (i - startIn)} + 1
                    endPre,          //前序右子树总长度: 前序的总长度
                    in,              //中序集合
                    i + 1,           //中序右子树的起始点:i+1
                    endIn);          //中序右子树总长度: 中序的总长度
            break;
        }
    }
    return node;
}

public static void main(String[] args){
    BinaryTree binaryTree= new BinaryTree();
    binaryTree.createBinaryTree();

    System.out.println("非递归前序");
    binaryTree.nonRecOrder(binaryTree.root);

    System.out.println("递归前序");
    binaryTree.preOrder(binaryTree.root);

    System.out.println("递归中序");
    binaryTree.midOrder(binaryTree.root);

    System.out.println("递归后序");
    binaryTree.postOrder(binaryTree.root);

    String[] pre = {"A","B","D","G","H","C","E","I","F"};

    String[] mid = {"G","D","H","B","A","E","I","C","F"};

    TreeNode root = binaryTree.resetBinaryTreeString(pre, 0, pre.length - 1, mid, 0, mid.length - 1);

    System.out.println("重建二叉树前序");
    binaryTree.preOrder(root);
}

public class TreeNode {
    /**
     *
     */
    private int index;
    /**
     * 数据域
     */
    private String data;
    /**
     * 左子树
     */
    private TreeNode leftChild;
    /**
     * 右子树
     */
    private TreeNode rightChild;


    public int getIndex() {
        return index;
    }


    public void setIndex(int index) {
        this.index = index;
    }


    public String getData() {
        return data;
    }


    public void setData(String data) {
        this.data = data;
    }


    public TreeNode(int index, String data) {
        this.index = index;
        this.data = data;
        this.leftChild = null;
        this.rightChild = null;
    }

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

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容

  • 树的概述 树是一种非常常用的数据结构,树与前面介绍的线性表,栈,队列等线性结构不同,树是一种非线性结构 1.树的定...
    Jack921阅读 4,432评论 1 31
  • 基于树实现的数据结构,具有两个核心特征: 逻辑结构:数据元素之间具有层次关系; 数据运算:操作方法具有Log级的平...
    yhthu阅读 4,235评论 1 5
  • 数据结构和算法--二叉树的实现 几种二叉树 1、二叉树 和普通的树相比,二叉树有如下特点: 每个结点最多只有两棵子...
    sunhaiyu阅读 6,422评论 0 14
  • 一直以来,我都很少使用也避免使用到树和图,总觉得它们神秘而又复杂,但是树在一些运算和查找中也不可避免的要使用到,那...
    24K男阅读 6,726评论 5 14
  • 这几天开学,学校还在上课,最近也是在找工作,很多天都没有更新文章,现在补一篇二叉树的文章。 最近校招公司的笔试陆续...
    zero_sr阅读 3,939评论 0 5