学习笔记-线索化二叉树

本文介绍了线索化二叉树的概念,通过在节点中增加leftType和rightType来区分子树和前后继节点。在中序线索化二叉树的过程中,空闲的指针被用来连接前驱和后继节点。中序遍历线索化二叉树时,采用多重循环,从左边界节点开始,沿着后继节点进行遍历。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

线索化二叉树

线索化二叉树的介绍:
在这里插入图片描述
对于这个普通二叉树来说,节点8,10,14的左右指针是空的,6的右指针是空的,也就是说并没有充分的利用到所有的指针。那么,我们可以对该树线索化,使得空闲的指针被充分利用。
在这里插入图片描述

中序线索化二叉树

在这里插入图片描述

对于一个线索二叉树(如上图),我们让空余节点的左指针指向它的前驱结点,右指针指向它的后继节点。这样一来,该节点的左指针有可能指向它的左子树,也可能指向前驱节点,该节点的右指针有可能指向它的右子树,也有可能指向后继节点。
所以,为了区分,我们在普通树节点的基础上设置leftType和rightType,当0时表示指向左右子树,1时表示指向前驱后继。

//线索化二叉树的节点
class Hero2{
    public int no;
    public String name;
    public Hero2 left;
    public Hero2 right;
    //0代表是左子树,1代表是前驱节点
    public int leftType;
    //0代表是右子树,1代表是后继节点
    public int rightType;

    public Hero2(int no, String name) {
        this.no = no;
        this.name = name;
    }

    @Override
    public String toString() {
        return "Hero2{" +
                "no=" + no +
                ", name='" + name + '\'' +
                '}';
    }
}

同时,因为二叉树是单向的,为了连接节点和前驱结点,我们在树中设置一个pre存储前驱节点

中序线索化思路:
1.如果节点是空的,就无法线索化,直接返回。
2.线索化节点的左子树,也就是向左递归。
3.线索化自己。
第一步,如果自己的左子节点为空,那么让左子节点连接上前驱结点pre,然后leftType就为1(当然,第一次时pre为null)。
第二步,如果右子节点为空,就将右子节点连接上后继结点。如上图,8要连上3,但因为树是单向的,8是不能直接访问3的,所以这一步操作实际上发生在下一次递归,在下一次的递归中,pre就指向8,而node指向3(中序遍历的顺序),使得8连上3,实际上是当pre!=null且pre的right为空时,让pre的right连上node并把rightType设为1。
最后一步,让pre=node,这就实现了pre的后移,pre最开始是null,随后依次指向中序遍历的每一个数。
4.线索化右子树,也就是向右递归。

//线索化二叉树
class ThreadedBinaryTree{
    public Hero2 root;
    //存储前驱节点
    public Hero2 pre=null;

    //中序线索化方法
    public void threadNodes(Hero2 node){
        if(node==null){
            return;
        }
        //线索化左子树
        threadNodes(node.left);
        //当前节点的前驱节点线索化
        if(node.left==null){
            node.left=pre;
            node.leftType=1;
        }
        //后继节点线索化,实际上是下一个递归
        if(pre!=null && pre.right==null){
            pre.right=node;
            pre.rightType=1;
        }
        //移动前驱节点
        pre=node;
        //线索化右子树
        threadNodes(node.right);
    }
}

中序遍历线索化二叉树

在这里插入图片描述

根据线索化二叉树的特性,中序遍历的方法与遍历普通二叉树不同。采用多重循环遍历,总的思路顺着后继节点遍历整棵树。

具体思路是:首先找到中序遍历的头,也就是二叉树中最左边的那个节点,它的特点是leftType=1,因此用while循环,node一开始等于root,当node不为空时循环,为了找到头,node的leftType=0时,一直后移node,直到node根据上图指向8,也就是中序遍历的头,输出node。接下来根据后继节点进行中序遍历,中序线索化时,后继节点实际上就是二叉树进行中序遍历的顺序,所以我们直接输出该节点的后继节点就行,那么后继节点的特点是node的rightType=1。所以进行while循环,如果node的rightType=1。证明该节点的right指向后继节点,那先移动到后继结点,再输出node。最后,如果node的rightType=0,说明node的right直接连接右子树,那就不能直接输出,而要重新判断,让node=node.right,进入到下一轮循环。

 //中序线索化遍历
    public void threadedList(){
        //从根节点开始
        Hero2 node=root;
        while (node!=null){
            while (node.leftType==0){
                node=node.left;
            }
            System.out.println(node);
            while (node.rightType==1){
                node=node.right;
                System.out.println(node);
            }
            node=node.right;
        }
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值