顺序存储二叉树及线索化二叉树Java实现

±---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------@TOC

一、基本概念

1、顺序存储二叉树
  从数据存储来看,数组存储方式和树的存储方式可以互相转换,即数组可以转化为树,树也可以转化为数组。
顺序存储二叉树的特点:

  1. 二叉树顺序存储通常只考虑完全二叉树。
  2. 第n个元素的左子节点为2*n+1
  3. 第n个元素的右子节点为2*n+1
  4. 第n个元素的父节点为(n-1)/2
  5. n表示二叉树中的第几个元素或数组下标(从0开始编号)
    如图所示:
    在这里插入图片描述

2、线索化二叉树
  对于n个结点的二叉树,在二叉链存储结构中有n+1【公式:2n-(n-1)=n+1】个空链域,利用这些空链域存放在某种遍历次序下该结点的前驱结点和后继结点的指针,这些指针称为线索,加上线索的二叉树称为线索二叉树。线索链表解决了无法直接找到该结点在某种遍历序列中的前驱和后继结点的问题,解决了二叉链表找左、右孩子困难的问题。

二、思路分析

1、顺序存储二叉树
  对于给定的一个顺序存储的二叉树,实际上就是一个数组,只是对于普通的数组,我们在遍历时通常会按照由低索引向高索引依次遍历,而顺序存储的二叉树由于要考虑到它的原型是一个完全二叉树,所以在遍历时,需要先计算出对应节点左右子节点的索引,再按照树的前中后序遍历方法进行遍历。
2、线索化二叉树
如图所示二叉树进行中序线索化。
二叉树
该二叉树的中序遍历序列为:[8,3,10,1,14,6],将其进行线索化为:即叶子节点会生成一个指向中序遍历的前驱节点的索引
在这里插入图片描述
当线索化二叉树后,Node节点的属性left和right,会出现如下两种情况:

  1. left指向的是左子树,也可能指向的是前驱节点。如节点1的left指向的是左子树,而10节点的left指向的则是前驱节点。
  2. right指向的是右子树,也可能指向的是后继节点。如节点1的right指向的是右子树,而节点10的right则指向的是后继节点。
    所以在实现时,我们需要来设置一个标记区分这两种情况。

注意:线索化二叉树的遍历和前面说到的二叉树的遍历有所不同,因为前面的二叉树遍历停止的条件的当左子树或右子树为空,而线索化二叉树多了指向前驱或者后继节点的指针,所以会出现死循环。所以对于线索化二叉树的遍历可以通过线性方式进行遍历,无需再使用递归遍历。

三、代码实现

1、顺序存储二叉树
创建树结构以及相关遍历方法

class ArrBinaryTree {
    private int[] arr;

    public ArrBinaryTree(int[] arr) {
        this.arr = arr;
    }

    //封装前中后序遍历方法
    public void preOrder() {
        this.preOrder(0);
    }

    public void midOrder() {
        this.midOrder(0);
    }

    public void postOrder() {
        this.postOrder(0);
    }

    /**
     * 前序遍历
     * @param index 根节点索引
     */
    private void preOrder(int index) {
        if (arr == null || arr.length == 0) {
            System.out.println("数组为空");
        }
        System.out.print(arr[index]);

        if ((index * 2 + 1) < arr.length) {
            preOrder(2 * index + 1);
        }
        if ((index * 2 + 2) < arr.length) {
            preOrder(2 * index + 2);
        }
    }

    /**
     * 中序遍历
     * @param index 根节点索引
     */
    private void midOrder(int index) {
        if (arr == null || arr.length == 0) {
            System.out.println("数组为空");
        }

        if ((index * 2 + 1) < arr.length) {
            midOrder(2 * index + 1);
        }

        System.out.print(arr[index]);

        if ((index * 2 + 2) < arr.length) {
            midOrder(2 * index + 2);
        }
    }

    /**
     * 后序遍历
     * @param index 根节点索引
     */
    private void postOrder(int index) {
        if (arr == null || arr.length == 0) {
            System.out.println("数组为空");
        }
        if ((index * 2 + 1) < arr.length) {
            postOrder(index * 2 + 1);
        }
        if ((index * 2 + 2) < arr.length) {
            postOrder(index * 2 + 2);
        }
        System.out.print(arr[index]);
    }
}

写一个demo来进行测试

public class ArrBinaryTreeDemo {
    public static void main(String[] args) {
        int[] arr = {1, 2, 3, 4, 5, 6};
        ArrBinaryTree tree = new ArrBinaryTree(arr);
        System.out.println("前序遍历:");
        tree.preOrder();
        System.out.println("\n中序遍历:");
        tree.midOrder();
        System.out.println("\n后序遍历:");
        tree.postOrder();

    }
}

测试结果:
测试
2、线索化二叉树
首先节点类

class Node {
    private int no;
    private String name;
    private Node left;
    private Node right;
    private int leftType;//leftType==0,表示指向左子树,leftType==1表示指向前驱节点
    private int rightType;//leftType==0,表示指向右子树,leftType==1表示指向后继节点

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

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

    public int getNo() {
        return no;
    }

    public void setNo(int no) {
        this.no = no;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Node getLeft() {
        return left;
    }

    public void setLeft(Node left) {
        this.left = left;
    }

    public Node getRight() {
        return right;
    }

    public void setRight(Node right) {
        this.right = right;
    }

    public int getLeftType() {
        return leftType;
    }

    public void setLeftType(int leftType) {
        this.leftType = leftType;
    }

    public int getRightType() {
        return rightType;
    }

    public void setRightType(int rightType) {
        this.rightType = rightType;
    }
}

然后创建一个线索化二叉树以及相关方法

class ThreadedBinaryTree {
    private Node root;

    //指向当前节点的前驱节点的指针
    //在递归进行线索化时,pre总是保留前一个节点
    private Node pre = null;

    public void setRoot(Node root) {
        this.root = root;
    }

    //编写线索化方法

    public void threadedNodes() {
        this.threadedNodes(root);
    }
    /**
     *
     * @param node  当前需要线索化的节点
     */
    public void threadedNodes(Node node) {
        //如果节点为空,则不能进行线索化
        if (node == null) {
            return;
        }
        //线索化左子树
        threadedNodes(node.getLeft());

        //线索化当前节点
        //处理当前节点的前驱节点
        if (node.getLeft() == null) {
            //让当前节点的左指针指向前驱节点
            node.setLeft(pre);
            //修改当前节点的左指针的类型,指向前驱节点
            node.setLeftType(1);
        }
        //处理后继节点
        if (pre != null && pre.getRight() == null) {
            pre.setRight(node);
            pre.setRightType(1);
        }

        pre = node;//没处理一个节点,让当前节点是下一个节点的前驱节点

        //线索化右子树
        threadedNodes(node.getRight());

    }

    //中序遍历
    public void midOrder() {
        //定义一个遍历存储当前节点
        Node node = root;
        while (node != null) {
            //循环找到leftType==1的节点
            while (node.getLeftType() == 0) {
                node = node.getLeft();
            }
            //打印当前节点
            System.out.println(node);
            //如果当前节点指向的是后继节点,就一直输出
            while (node.getRightType() == 1) {
                //获取当前节点的后继结点
                node = node.getRight();
                System.out.println(node);
            }
            //替换这个遍历的节点
            node = node.getRight();
        }
    }
}

最后写一个demo来测试进行测试

public class ThreadedBinaryTreeDemo {
    public static void main(String[] args) {
        Node root = new Node(1, "Tom");
        Node node2 = new Node(3, "Jack");
        Node node3 = new Node(6, "smith");
        Node node4 = new Node(8, "mary");
        Node node5 = new Node(10, "king");
        Node node6 = new Node(14, "dim");

        root.setLeft(node2);
        root.setRight(node3);
        node2.setLeft(node4);
        node2.setRight(node5);
        node3.setLeft(node6);

        ThreadedBinaryTree tree = new ThreadedBinaryTree();
        tree.setRoot(root);
        tree.threadedNodes();

        //以节点10测试
        Node leftNode = node5.getLeft();
        Node rightNode = node5.getRight();
        System.out.println("10号节点的前驱节点是:" + leftNode);
        System.out.println("10号节点的后继节点是:" + rightNode);

        System.out.println("中序遍历线索化二叉树:");
        tree.midOrder();

    }
}

测试结果
测试结果

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值