树形结构——JAVA实现

news2025/1/9 1:39:07

1、树定义和基本术语

节点

package com.young.tree;

/**
 * <p>
 * Title:树节点:二叉链表结构
 * </p>
 *
 * @Author: yangyongbing
 * @Date: 2023-04-18 13:25
 * @version: v1.0
 */
public class Node<T> {

    public Node<T> lChild;
    private T data;
    public Node<T> rChild;

    public Node() {
        lChild=null;
        data=null;
        rChild=null;
    }

    public Node(T data) {
        this.data = data;
        this.lChild=null;
        this.rChild=null;
    }

    public Node<T> getlChild() {
        return lChild;
    }

    public void setlChild(Node<T> lChild) {
        this.lChild = lChild;
    }

    public T getData() {
        return data;
    }

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

    public Node<T> getrChild() {
        return rChild;
    }

    public void setrChild(Node<T> rChild) {
        this.rChild = rChild;
    }
}

二叉树

package com.young.tree;

/**
 * <p>
 * Title: 二叉树
 * </p>
 *
 * @Author: yangyongbing
 * @Date: 2023-04-18
 * @version: v1.0
 */
public class BinaryTree<T> {

    private final int maxNodes = 100;

    // 根节点
    public Node<T> root;

    // 创建一棵空二叉树
    public BinaryTree() {
        this.root = new Node<>();
    }

    // 创建一棵以数据元素x为根节点的二叉树
    public BinaryTree(T x) {
        this.root = new Node<>(x);
    }

    /*在当前二叉树的parent结点中插入一个新的左子结点,
   若已存在左子树,则将该左子树变成新左子结点的左孩子树*/
    public boolean addLeft(T x, Node<T> parent) {
        if (parent == null) {
            return false;
        }

        // 创建一个空节点
        Node<T> p = new Node<>(x);
        // 如果父节点的左子树为空,则直接将数据素x赋给父节点的左孩子节点
        if (parent.lChild != null) {
            // 将父节点的左子树赋给这个新节点左子节点
            p.lChild = parent.lChild;
        }
        // 将新节点赋给父节点的左孩子节点
        parent.lChild = p;

        return true;
    }

    /*在当前二叉树的parent结点中插入一个新的右子节点,
    若已存在右子树,则将该右子树变成新右子节点的右孩子树*/
    public boolean addRight(T x, Node<T> parent) {
        if (parent == null) {
            return false;
        }
        // 创建一个空节点
        Node<T> p = new Node<>(x);
        if (parent.rChild != null) {
            p.rChild = parent.rChild;
        }
        parent.rChild = p;
        return true;
    }

    // 删除当前二叉树的parent节点中的左子树
    public boolean deleteLeft(Node<T> parent) {
        if (parent == null) {
            return false;
        } else {
            parent.lChild = null;
            return true;
        }
    }

    // 删除当前二叉树的parent节点中的右子树
    public boolean deleteRight(Node<T> parent) {
        if (parent == null) {
            return false;
        } else {
            parent.rChild = null;
            return true;
        }
    }

    // 先序遍历
    public void preorder(Node<T> node) {
        if (node != null) {
            // 访问根节点
            visit(node.getData());
            // 先序遍历左子树
            preorder(node.getlChild());
            // 先序遍历右子树
            preorder(node.getrChild());
        }
    }

    // 中序遍历
    public void inorder(Node<T> node) {
        if (node != null) {
            // 中序遍历左子树
            inorder(node.lChild);
            // 访问根节点
            visit(node.getData());
            // 中序遍历右子树
            inorder(node.rChild);
        }
    }

    // 后序遍历
    public void postorder(Node<T> node) {
        if (node != null) {
            // 后续遍历左子树
            postorder(node.lChild);
            // 后续遍历右子树
            postorder(node.rChild);
            // 访问根节点
            visit(node.getData());
        }
    }

    // 按层次遍历
    public void levelOrder() {
        // 节点数组,用于存放节点
        Node<T>[] queue = new Node[this.maxNodes];
        if (this.root == null) {
            return;
        }
        // 定义队首和队尾指针
        int front, rear;
        // 队列为空,对首指针不指向任何一个数组元素
        front = -1;
        // 队列为空,对尾指针指向数组第一个位置
        rear = 0;
        // 根节点入队
        queue[rear] = this.root;
        while (front != rear) {
            // 访问对首节点的数据域
            front++;
            visit(queue[front].getData());
            // 左节点入队
            if (queue[front].lChild != null) {
                rear++;
                queue[rear] = queue[front].lChild;
            }
            // 右节点入队
            if (queue[front].rChild != null) {
                rear++;
                queue[rear] = queue[front].rChild;
            }
        }

    }

    private void visit(T x) {
        System.out.println(x);
    }

    // 在当前二叉树中查找数据x
    public boolean search(Node<T> node,T x) {
        if (node != null) {
            // 访问根节点
            T t = node.getData();
            if(x==t){
                return true;
            }
            // 先序遍历左子树
            search(node.getlChild(),x);
            // 先序遍历右子树
            search(node.getrChild(),x);
        }
        return false;
    }

    // 按某种方式遍历二叉树中的所有节点
    //按指定方式遍历二叉树
    //i=0表示先序遍历,=1表示中序遍历,=2表示后序遍历,=3表示层次遍历
    public void traversal(int i)
    {
        switch(i)
        {
            case 0: preorder(this.root);break;
            case 1: inorder(this.root);break;
            case 2: postorder(this.root);break;
            default: levelOrder();
        }
    }
    // 求当前二叉树的高度
    public int getHeight(Node<T> parent) {

        int lh,rh,max;
        if(parent!=null){
            lh=getHeight(parent.lChild);
            rh=getHeight(parent.rChild);
            max= Math.max(lh, rh);
            return max+1;
        }
        return 0;
    }
}

1.1、树定义

树(Tree)是若干个结点组成的有限集合,其中必须有一个结点是根结点,其余结点划分为若干个互不相交的集合,每一个集合还是一棵树,但被称为根的子树。注意,当树的结点个数为0时,我们称这棵树为空树,记为Φ。
特点:

  • 树的根结点没有前驱结点,除根结点之外的所有结点有且只有一个前驱结点。
  • 树中所有结点可以有零个或多个后继结点。

1.2、术语

  • 结点:表示树中的元素,包括数据项及若干指向其子树的分支。
  • 结点的度:结点所拥有的子树的个数称为该结点的度。
  • 叶子结点:度为0的结点称为叶子结点,或者称为终端结点。
  • 分支结点:度不为0的结点称为分支结点,或者称为非终端结点。一棵树的结点除叶子结点外,其余的都是分支结点。
  • 孩子、双亲、兄弟:若在树中一个结点A的子树的根结点是B,则称B为A的孩子(也称子结点),称A为B的双亲(也称父节点)。具有同一个双亲的子结点互称为兄弟。
  • 路径、路径长度:如果一棵树的一串结点n1,n2,…,nk有如下关系,即结点ni是ni+1的父结点(1≤i<k),就把n1,n2,…,nk称为一条由n1至nk的路径。这条路径的长度是k-1。
  • 祖先、子孙:在树中,如果有一条路径从结点M到结点N,那么M就称为N的祖先,而N称为M的子孙。
  • 结点的层数:规定树的根结点的层数为1,其余结点的层数等于它的双亲结点的层数加1。
  • 树的深度:树中所有结点的最大层数称为树的深度.
  • 树的度:树中各结点度的最大值称为该树的度。
  • 有序树和无序树:如果一棵树中结点的各子树从左到右是有次序的,即若交换了某结点各子树的相对位置,则构成不同的树,称这棵树为有序树;反之,则称为无序树。
  • 森林:零棵或有限棵不相交的树的集合称为森林。自然界中树和森林是不同的概念,但在数据结构中,树和森林只有很小的差别。任何一棵树,删去根结点就变成了森林。

2、二叉树

2.1、二叉树基本概念

  • 二叉树
    二叉树(Binary Tree)是一种每结点最多拥有2个子树的树结构,其中第1个子树被称为左子树,第2个子树被称为右子树。注意,当二叉树的结点个数为0时,我们称这个二叉树为空二叉树,记为Φ。二叉树是有序的,即若将其左、右子树颠倒,就成为另一棵不同的二叉树。即使树中结点只有一棵子树,也要区分它是左子树,还是右子树。因此二叉树具有五种基本形态。
    在这里插入图片描述

  • 满二叉树
    在一棵二叉树中,如果所有分支结点都存在左子树和右子树,并且所有叶子结点都在同一层上,这样的一棵二叉树称作满二叉树。(a)图就是一棵满二叉树,(b)图则不是满二叉树,因为该二叉树的D,F,G,H,I叶子结点未在同一层上。
    在这里插入图片描述

  • 完全二叉树
    完全二叉树是一种叶子结点只能出现在最下层和次下层且最下层的叶子结点集中在树的左边的特殊二叉树。图5.5(a)所示为一棵完全二叉树,图5.4(b)和图5.5(b)都不是完全二叉树。对比图5.4(a)和图5.5(a)可以发现,满二叉树与完全二叉树存在如下关系:当树的深度相同时,若对树的结点按从上至下、从左到右的顺序进行编号,则在两种树上同一个位置上的结点的编号相同。显然,一棵满二叉树必定是一棵完全二叉树,而完全二叉树未必是满二叉树。
    在这里插入图片描述

  • 二叉树性质
    性质1 一棵非空二叉树的第i层上最多有2i-1个结点(i≥1)。
    性质2 一棵深度为k的二叉树中,最多具有2k-1个结点。
    性质3 对于一棵非空的二叉树,如果叶子结点数为n0,度数为2的结点数为n2,则有:n0=n2+1。
    性质4 具有n个结点的完全二叉树的深度k为[Log2n]
    性质5 对于具有n个结点的完全二叉树,如果按照从上至下和从左到右的顺序对二叉树中的所有结点从1开始顺序编号,则对于任意的序号为i的结点,有:
    (1)如果i>1,则序号为i的结点的父结点的序号为[插图];如果i=1,则该结点是根结点,无父结点。
    (2)如果2i≤n,则序号为i的结点的左子结点的序号为2i;如果2i>n,则序号为i的结点无左子结点。
    (3)如果2i+1≤n,则序号为i的结点的右子结点的序号为2i+1;如果2i+1>n,则序号为i的结点无右子结点。+1。

2.2、二叉树的存储结构

  • 顺序存储结构
    所谓二叉树的顺序存储,就是用一组连续的存储单元存放二叉树中的结点。一般是按照二叉树结点从上至下、从左到右的顺序存储。这样结点在存储位置上的前驱、后继关系并不一定就是它们在逻辑上的邻接关系,然而只有通过一些方法确定某结点在逻辑上的前驱结点和后继结点,这种存储才有意义。因此,依据二叉树的性质,完全二叉树和满二叉树采用顺序存储比较合适,树中结点的序号可以唯一地反映出结点之间的逻辑关系,这样既能够最大可能地节省存储空间,又可以利用数组元素的下标值确定结点在二叉树中的位置,以及结点之间的关系。
    在这里插入图片描述
    在这里插入图片描述
  • 链式存储结构
    所谓二叉树的链式存储结构是指用链表来表示一棵二叉树,即用链来指示元素的逻辑关系。通常有下面两种形式。
    (1) 二叉链表存储
    链表中每个结点由三个域组成,除了数据域外,还有两个指针域,分别用来给出该结点左孩子和右孩子所在的链结点的存储地址。结点的存储结构为:
    在这里插入图片描述
    其中,data域存放某结点的数据信息;lchild与rchild分别存放指向左孩子和右孩子的指针,当左孩子或右孩子不存在时,相应指针域值为空(用符号∧或null表示)。
    二叉树链式存储的每个结点可描述为:
class Node<T>
{
        public Node<T> lChild;     //左孩子
        private T data;             //数据域
        public Node<T> rChild;     //右孩子
        public Node()              //构造函数,创建一个空节点
        {
                data = null;
                lChild = null;
                rChild = null;
        }
        public Node(T x)               //重载构造函数,创建一个数据值为x的节点
        {
                data = x;
                lChild = null;
                rChild = null;
        }
}

二叉链表也可以带头结点的方式存放:
在这里插入图片描述
在Java中描述二叉链表的关键是确定二叉树的根,代码如下。

class BinaryTree<T>
{
        public Node<T> root;       //根结点
        public BinaryTree()  //创建一棵空二叉树
        {
                 this.root = new Node<T>();
        }
        public BinaryTree(T x)   //创建一棵以数据元素x为根结点的二叉树
        {
                this.root = new Node<T>(x);
         }
    //……
}

(2)三叉链表存储:
每个结点由四个域组成,具体结构为:
在这里插入图片描述
其中,data、lchild以及rchild三个域的意义同二叉链表结构,parent域为指向该结点双亲结点的指针。这种存储结构既便于查找孩子结点,又便于查找双亲结点,但是,相对于二叉链表存储结构而言,它增加了空间开销。
一棵二叉树的三叉链表表示:
在这里插入图片描述
尽管在二叉链表中无法由结点直接找到其双亲,但由于二叉链表结构灵活,操作方便,对于一般情况的二叉树,甚至比顺序存储结构还节省空间。因此,二叉链表是最常用的二叉树存储方式。

2.3、二叉树的基本操作及实现

  • 二叉树的基本操作:
class BinaryTree<T>
{
    private Node<T> root;
    public BinaryTree(){}      //创建一棵空二叉树
    public BinaryTree(T x){}   //创建一棵以数据元素x为根结点的二叉树
       /*在当前二叉树的parent结点中插入一个新的左子结点,
若已存在左子树,则将该左子树变成新左子结点的左孩子树*/
    public boolean insertLeft(T x, Node<T> parent){ }
       /*在当前二叉树的parent结点中插入一个新的右孩子结点,
若已存在右子树,则将该右子树变成新右孩子结点的左子树*/
    public boolean insertRight(Node<T> parent){ }
    //删除在当前二叉树的parent结点中的左子树
    public boolean deleteLeft(Node<T> parent){ }
    //删除在当前二叉树的parent结点中的右子树
    public boolean deleteRight(Node<T> parent){ }
    public boolean search(T x){ } //在当前二叉树中查找数据x
    public void traversal(int i){ }  //按某种方式遍历当前二叉树的全部结点
    public int getHeight(Node<T> parent){ }   //求当前二叉树的高度
}
  • 算法的实现:
    算法的实现依赖于具体的存储结构,当二叉树采用不同的存储结构时,上述各种操作的实现算法是不同的。下面讨论基于二叉链表存储结构的上述操作的实现算法。
    建立一棵空二叉树:
public BinaryTree()
{
        this.root = new Node<T>();  //创建根结点,该结点的数据域为空
}

生成一棵二叉树:

public BinaryTree(T x)   //创建一棵以数据元素x为根结点的二叉树
{
        this.root = new Node<T>(x);
}

向二叉树中插入一个左孩子结点:

//在当前二叉树的parent节点中插入一个新的左孩子结点,若已存在左子树,则将该左子树变成新左孩子结点的左子树
public boolean insertLeft(T x, Node<T> parent)
{
        if(parent==null)        return false;
        Node<T> p= new Node<T>(x);    //创建一个新结点
        if(parent.lChild==null)
                parent.lChild = p;   //将新结点直接设置到父结点的左孩子结点
        else
        {
                //先将父结点原来的左子树设置为新结点的左子树
                p.lChild = parent.lChild;
                //再将新结点设置到父结点的左孩子结点
                parent.lChild = p;
        }
        return true;
}
//注意,若要执行本操作,则必须先确定插入位置,即parent节点

删除二叉树的左子树:

//删除当前二叉树的parent结点中的左子树
public boolean deleteLeft(Node<T> parent)
{
        if(parent==null)        return false;
        else
        {
                parent.lChild=null;
                return true;
        }
}

2.4、二叉树的遍历

二叉树的遍历是指按照某种顺序访问二叉树中的每个结点,使每个结点被访问一次且仅被访问一次。
遍历是二叉树中经常要用到的一种操作。因为在实际应用问题中,常常需要按一定顺序对二叉树中的每个结点逐个进行访问,查找具有某一特点的结点,然后对这些满足条件的结点进行处理。
通过一次完整的遍历,可使二叉树中结点信息由非线性排列变为某种意义上的线性序列。也就是说,遍历操作使非线性结构线性化。
由二叉树的定义可知,一棵二叉树由根结点、根结点的左子树和根结点的右子树三部分组成。因此,只要依次遍历这三部分,就可以遍历整个二叉树。若以D、L、R分别表示访问根结点、遍历根结点的左子树、遍历根结点的右子树,则二叉树的遍历方式有六种:DLR、LDR、LRD、DRL、RDL和RLD。如果限定先左后右,则只有前三种方式,即DLR(称为先序遍历)、LDR(称为中序遍历)和LRD(称为后序遍历)。

  • 先序遍历(DLR)
    先序遍历的递归过程为:
    若二叉树为空,遍历结束,否则,
    (1)访问根结点;
    (2)先序遍历根结点的左子树;
    (3)先序遍历根结点的右子树。
public void preorder(Node<T> node)
{
        if(node==null) return;
        else
        {
                visit(node.getData());   //访问根结点
                preOrder(node.lChild);   //先序遍历左子树
                preOrder(node.rChild);   //先序遍历右子树
        }
}
  • 中序遍历(LDR)
    中序遍历的递归过程为:
    若二叉树为空,遍历结束,否则,
    (1)中序遍历根结点的左子树;
    (2)访问根结点;
    (3)中序遍历根结点的右子树。
public void inorder(Node<T> node)
{
        if(node==null) return;
        else
        {
                inorder(node.lChild);   //中序遍历左子树
                visit(node.getData());  //访问根结点
                inorder(node.rChild);   //中序遍历右子树
        }
}
  • 后序遍历(LRD)
    后序遍历的递归过程为:
    若二叉树为空,遍历结束,否则,
    (1)后序遍历根结点的左子树;
    (2)后序遍历根结点的右子树;
    (3)访问根结点。
public void postorder(Node<T> node)
{
        if(node==null) return;
        else
        {
                postorder(node.lChild);   //后序遍历左子树
                postorder(node.rChild);   //后序遍历右子树
                visit(node.getData());  //访问根结点
        }
}
  • 层次遍历
    二叉树的层次遍历,是指从二叉树的第一层(根结点)开始,从上至下逐层遍历,在同一层中,则按从左到右的顺序对结点逐个访问。
    由层次遍历的定义可以推知,在进行层次遍历时,对一层结点访问完后,再按照它们的访问次序对各个结点的左孩子和右孩子顺序访问,这样一层一层进行,先遇到的结点先访问,这与队列的操作原则比较吻合。因此,在进行层次遍历时,可设置一个队列结构,遍历从二叉树的根结点开始,首先将根结点进队列,然后从队头取出一个元素,每取一个元素,执行下面两个操作。
    (1)访问该元素所指结点。
    (2)若该元素所指结点的左、右孩子结点非空,则将该元素所指结点的左孩子结点和右孩子结点顺序进队。此过程不断进行,当队列为空时,二叉树的层次遍历结束。
public void levelOrder()
{
        Node<T>[] queue= new Node[this.maxNodes];//构造一个队列
        int front,rear;    //队首指针、队尾指针
        if (this.root==null) return;
        front=-1;      //队列暂时为空,队首指针不指向任何一个数组元素
        rear=0;        //队列暂时为空,队尾指针指向第一个数组元素
        queue[rear]=this.root;  //二叉树的根结点进队列
        while(front!=rear)
        {
                front++;
                visit(queue[front].getData());    /*访问队首结点的数据域*/
                /*将队首结点的左孩子结点进队列*/
                if (queue[front].lChild!=null)
                {
                        rear++;
                        queue[rear]=queue[front].lChild;
                }
                /*将队首结点的右孩子结点进队列*/
                if (queue[front].rChild!=null)
                {
                        rear++;
                        queue[rear]=queue[front].rChild;
                }
        }
}

2.5、完整代码

package com.young.tree;

/**
 * <p>
 * Title:树节点:二叉链表结构
 * </p>
 *
 * @Author: yangyongbing
 * @Date: 2023-04-18 13:25
 * @version: v1.0
 */
public class Node<T> {

    public Node<T> lChild;
    private T data;
    public Node<T> rChild;

    public Node() {
        lChild=null;
        data=null;
        rChild=null;
    }

    public Node(T data) {
        this.data = data;
        this.lChild=null;
        this.rChild=null;
    }

    public Node<T> getlChild() {
        return lChild;
    }

    public void setlChild(Node<T> lChild) {
        this.lChild = lChild;
    }

    public T getData() {
        return data;
    }

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

    public Node<T> getrChild() {
        return rChild;
    }

    public void setrChild(Node<T> rChild) {
        this.rChild = rChild;
    }
}

package com.young.tree;

/**
 * <p>
 * Title: 二叉树
 * </p>
 *
 * @Author: yangyongbing
 * @Date: 2023-04-18
 * @version: v1.0
 */
public class BinaryTree<T> {

    private final int maxNodes = 100;

    // 根节点
    public Node<T> root;

    // 创建一棵空二叉树
    public BinaryTree() {
        this.root = new Node<>();
    }

    // 创建一棵以数据元素x为根节点的二叉树
    public BinaryTree(T x) {
        this.root = new Node<>(x);
    }

    /*在当前二叉树的parent结点中插入一个新的左子结点,
   若已存在左子树,则将该左子树变成新左子结点的左孩子树*/
    public boolean addLeft(T x, Node<T> parent) {
        if (parent == null) {
            return false;
        }

        // 创建一个空节点
        Node<T> p = new Node<>(x);
        // 如果父节点的左子树为空,则直接将数据素x赋给父节点的左孩子节点
        if (parent.lChild != null) {
            // 将父节点的左子树赋给这个新节点左子节点
            p.lChild = parent.lChild;
        }
        // 将新节点赋给父节点的左孩子节点
        parent.lChild = p;

        return true;
    }

    /*在当前二叉树的parent结点中插入一个新的右子节点,
    若已存在右子树,则将该右子树变成新右子节点的右孩子树*/
    public boolean addRight(T x, Node<T> parent) {
        if (parent == null) {
            return false;
        }
        // 创建一个空节点
        Node<T> p = new Node<>(x);
        if (parent.rChild != null) {
            p.rChild = parent.rChild;
        }
        parent.rChild = p;
        return true;
    }

    // 删除当前二叉树的parent节点中的左子树
    public boolean deleteLeft(Node<T> parent) {
        if (parent == null) {
            return false;
        } else {
            parent.lChild = null;
            return true;
        }
    }

    // 删除当前二叉树的parent节点中的右子树
    public boolean deleteRight(Node<T> parent) {
        if (parent == null) {
            return false;
        } else {
            parent.rChild = null;
            return true;
        }
    }

    // 先序遍历
    public void preorder(Node<T> node) {
        if (node != null) {
            // 访问根节点
            visit(node.getData());
            // 先序遍历左子树
            preorder(node.getlChild());
            // 先序遍历右子树
            preorder(node.getrChild());
        }
    }

    // 中序遍历
    public void inorder(Node<T> node) {
        if (node != null) {
            // 中序遍历左子树
            inorder(node.lChild);
            // 访问根节点
            visit(node.getData());
            // 中序遍历右子树
            inorder(node.rChild);
        }
    }

    // 后序遍历
    public void postorder(Node<T> node) {
        if (node != null) {
            // 后续遍历左子树
            postorder(node.lChild);
            // 后续遍历右子树
            postorder(node.rChild);
            // 访问根节点
            visit(node.getData());
        }
    }

    // 按层次遍历
    public void levelOrder() {
        // 节点数组,用于存放节点
        Node<T>[] queue = new Node[this.maxNodes];
        if (this.root == null) {
            return;
        }
        // 定义队首和队尾指针
        int front, rear;
        // 队列为空,对首指针不指向任何一个数组元素
        front = -1;
        // 队列为空,对尾指针指向数组第一个位置
        rear = 0;
        // 根节点入队
        queue[rear] = this.root;
        while (front != rear) {
            // 访问对首节点的数据域
            front++;
            visit(queue[front].getData());
            // 左节点入队
            if (queue[front].lChild != null) {
                rear++;
                queue[rear] = queue[front].lChild;
            }
            // 右节点入队
            if (queue[front].rChild != null) {
                rear++;
                queue[rear] = queue[front].rChild;
            }
        }

    }

    private void visit(T x) {
        System.out.println(x);
    }

    // 在当前二叉树中查找数据x
    public boolean search(Node<T> node,T x) {
        if (node != null) {
            // 访问根节点
            T t = node.getData();
            if(x==t){
                return true;
            }
            // 先序遍历左子树
            search(node.getlChild(),x);
            // 先序遍历右子树
            search(node.getrChild(),x);
        }
        return false;
    }

    // 按某种方式遍历二叉树中的所有节点
    //按指定方式遍历二叉树
    //i=0表示先序遍历,=1表示中序遍历,=2表示后序遍历,=3表示层次遍历
    public void traversal(int i)
    {
        switch(i)
        {
            case 0: preorder(this.root);break;
            case 1: inorder(this.root);break;
            case 2: postorder(this.root);break;
            default: levelOrder();
        }
    }
    // 求当前二叉树的高度
    public int getHeight(Node<T> parent) {

        int lh,rh,max;
        if(parent!=null){
            lh=getHeight(parent.lChild);
            rh=getHeight(parent.rChild);
            max= Math.max(lh, rh);
            return max+1;
        }
        return 0;
    }
}

3、线索二叉树

3.1、定义

按照某种遍历方式对二叉树进行遍历,可以把二叉树中所有结点排列为一个线性序列。在该序列中,除第一个结点外,每个结点有且仅有一个直接前驱结点,除最后一个结点外,每个结点有且仅有一个直接后继结点。但是,二叉树中每个结点在这个序列中的直接前驱结点和直接后继结点是什么,二叉树的存储结构中并没有反映出来,只能在对二叉树遍历的动态过程中得到这些信息。为了保留结点在某种遍历序列中直接前驱和直接后继的位置信息,可以利用二叉树的二叉链表存储结构中的那些空指针域来指示。这些指向直接前驱结点和指向直接后继结点的指针被称为线索(thread),加了线索的二叉树称为线索二叉树。线索二叉树将为二叉树的遍历提供许多方便。

3.2、线索二叉树的结构

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/462660.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CASP15 蛋白质结构域 Domain 的定义和分类

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://blog.csdn.net/caroline_wendy/article/details/130379447 在CASP中&#xff0c;蛋白质结构域(Domain)的类别&#xff0c;包括 FM、FM/TBM、TBM-easy、TBM-hard、not evaluated 等5个类…

25从零开始学Java之数组扩容与数组拷贝的实现过程与原理分析

作者&#xff1a;孙玉昌&#xff0c;昵称【一一哥】&#xff0c;另外【壹壹哥】也是我哦 千锋教育高级教研员、CSDN博客专家、万粉博主、阿里云专家博主、掘金优质作者 前言 在上一篇文章中&#xff0c;壹哥给大家讲解了数组的创建、初始化及遍历方式&#xff0c;这些是我们学…

Cookies和Session案例-注册

1. 注册功能改进 1.1 service 将之前的注册案例的代码进行优化&#xff0c;将获取sqlsession工厂对象、获取sqlsession、获取mapper等操作从servlet中分离出来转变为三层架构的形式 在service目录下创建UserService public class UserService {SqlSessionFactory sqlSessionFa…

办公室组网

1.办公网络组网中,汇聚交换机和接入交换机你会做哪些配置? 接入交换机上配置: (1)VLAN配置:由题意得办公网络中有两个部门,使用VLAN技术将每个部门划入一个局域网中,如部门1属于VLAN 10,部门2属于VLAN20.该网络中还需要额外创建一个VLAN用于管理网络,如VLAN30。在接入…

一文了解国外AIGC头部产品

AIGC是指通过人工智能技术生成的内容&#xff0c;包括文字、图片、音频和视频等。AIGC技术可以基于大量的数据和算法&#xff0c;自动地生成各种类型的内容&#xff0c;可以用于新闻报道、广告宣传、文学创作、游戏设计等各个领域。AIGC技术的优点在于可以大大提高内容生产的效…

GD32F303RCT6开发笔记(一)—— macos环境搭建

macOS vscodegccpyocd环境搭建 1、vscode/arm-none-eabi-/pyocd 安装可百度。 2、pyocd 安装完成后&#xff0c;连接st-link 输入命令后显示如下&#xff0c;说明连接成功。 3、输入命令 pyocd pack find GD32F303RC4、如果没有安装GD32F303RC包 使用命令安装 pyocd pack …

【Vue 基础】vue-cli初始化项目及相关说明

目录 1. 创建项目 2. 项目文件介绍 3. 项目的其它配置 3.1 项目运行时&#xff0c;让浏览器自动打开 3.2 关闭eslint校验功能 3.3 src文件夹简写方法 1. 创建项目 vue create 项目名 2. 项目文件介绍 创建好的项目中包含如下文件&#xff1a; &#xff08;1&#xff09…

基于显扬科技自主研发3D机器视觉HY-M5在易拉罐包装检测的应用

行业现状&#xff1a; 易拉罐包装行业发展迅速&#xff0c;是中国食品工业的重要组成部分。近年来&#xff0c;随着经济水平的提高和生活方式变化&#xff0c;各类预包装食品需求剧增&#xff0c;碳酸饮料和啤酒等饮料消费大幅增加&#xff0c;直接带动易拉罐包装行业高速发展…

方案解析丨数字人主播如何成为电商直播新标配

浙江省政府办公厅近日印发《关于进一步扩大消费促进高质量发展若干举措》支持电子商务直播发展。抢抓电子商务直播快速发展机遇&#xff0c;发展数字人虚拟主播、元宇宙新消费场景等新业态新模式。 随着电商直播快速发展&#xff0c;企业怎么高效地实现引流获客&#xff0c;成为…

【计算机组成原理】数据的表示和运算·进位计数制

&#x1f6a9; 本文已收录至专栏&#xff1a;计算机基础 我们可以通过显示屏看到各种形式的数据信息&#xff0c;但数据是如何在计算机中表示呢&#xff1f;运算器又是如何实现数据的算数、逻辑运算&#xff1f; 十进制数是最适合我们日常使用的一种计数方式&#xff0c;除此之…

随手记录:Livox 时间戳修改为ROS时间戳

参考与前言 传感器类型&#xff1a;Livox-Mid70 参考链接&#xff1a;Ubuntu20.04系统安装Livox ROS Driver 官方驱动&#xff1a;https://github.com/Livox-SDK/livox_ros_driver 碎碎念&#xff1a;之所以要改成rostime主要是 提取pcd的时候发现这个timestamp 300.xxx 打…

我那张被问爆了的漫画头像确实有点东西

不得不说&#xff0c;这个照片变漫画有点东西啊&#xff01;不知道姐妹们有没有发现❗️最近漫画人像它又火起来了&#xff0c;基本上在dy等各大社交软件上&#xff0c;特别是朋友圈已经是刷屏了&#xff5e;随便一张照片经过渲染之后秒变动漫风格照&#xff0c;我就马不停蹄就…

宝塔面板搭建自己的网站,并发布公网远程访问

文章目录 1. 环境安装2. 安装cpolar内网穿透3. 内网穿透4.固定http地址5. 配置二级子域名6.创建一个测试页面 宝塔面板简单几步搭建本地web站点&#xff0c;并做内网穿透&#xff0c;实现公网用户也可以正常远程访问&#xff0c;无需公网IP&#xff0c;无需设置路由器。 1. 环…

这样的速度,还有谁?一个 issue 引发的性能大跃进

前段时间开源了一个关于音频特征提取和分析的小项目&#xff0c;自己是 AI 音频领域方向的&#xff0c;但受限于对音频特征的理解&#xff0c;做研究时总感觉缺乏“底料”&#xff0c;所以当做是学习练手做了这个小东西。 虽然是学习练手的小项目&#xff0c;但也信心满满&…

从盒马来看新零售的全面可行性

来源&#xff5c;新零售 不久前&#xff0c;一家位置极佳的北京老牌超市闭店的消息引发了很多人的唏嘘&#xff0c;这家超市位于北京长安街东侧的万达广场上&#xff0c;曾经作为万达广场的主力店&#xff0c;服务周边居民长达十年之久。 不过&#xff0c;周边的居民很快得知…

哪个牌子的电视盒子好用?经销商总结目前性能最好的电视盒子

做数码经销已经是第九年了&#xff0c;这些年对数码行业也算是颇有研究&#xff0c;大家选购数码产品时都会参考我的建议。今天我将来分享目前性能最好的电视盒子推荐&#xff0c;想知道哪个牌子的电视盒子好用看这篇就足够了。 一&#xff1a;泰捷WEBOX60Pro电视盒子 亮点&a…

27- OCR 光功率计数码管字符识别

要点&#xff1a; 光功率计数码管 1 前言 本案例将使用OCR技术自动识别光功率计显示屏文字&#xff0c;通过本章您可以掌握&#xff1a; PaddleOCR快速使用数据合成方法数据挖掘方法基于现有数据微调 为实现智能读数&#xff0c;通常会采取文本检测文本识别的方案&#xff…

【JavaWeb】jQuery(上)

本章内容 1.jQuery Hello world 2.jQuery 选择器 3.jQuery 过滤器 4.jQuery 元素筛选 1、jQuery 介绍 什么是 jQuery ? jQuery&#xff0c;顾名思义&#xff0c;也就是 JavaScript 和查询&#xff08;Query&#xff09;&#xff0c;它就是辅助 JavaScript 开发的 js 类…

ChatGPT能用来写小说吗-gpt可以续写小说吗

怎么用ChatGPT写网文 ChatGPT是一个语言生成模型&#xff0c;可以用于生成各种文本&#xff0c;包括网文。下面是一些写网文的建议。 确定你的主题和情节。在开始写作之前&#xff0c;你需要确保你有一个明确的主题和情节&#xff0c;这可以帮助你更好地组织你的故事&#xff0…

React18开发中遇到的一些小问题

遇到这样一个问题&#xff0c;初始化时用户登陆后需要获取到用户信息&#xff0c;但是发现获取用户信息这个接口触发了2次&#xff0c;这是不应该的&#xff0c;于是我查阅了一下资料&#xff0c;把自己的笔记记录下来。 还有就是使用mobx遇到的控制台警告问题&#xff0c;也一…