数据结构之树(二分搜索树)

简介: 数据结构之树(二分搜索树)

为什么要使用树结构


网络异常,图片无法展示
|


二叉树


网络异常,图片无法展示
|


网络异常,图片无法展示
|


二分搜索树


  • 存储的元素都必须具有可比较性。


网络异常,图片无法展示
|


二分搜索树的设计


// 定义内部类,设置节点
    private class Node {
        public E e;
        public Node left, right;
        public Node(E e) {
            this.e = e;
            left = null;
            right = null;
        }
    }
    // 根节点
    private Node root;
    private int size;


插入元素


不考虑重复元素的插入


网络异常,图片无法展示
|


这个是比较容易理解的递归


网络异常,图片无法展示
|


下面是更简洁的递归算法,不需要判断根节点是否为空。注意,这里返回的node是每次遍历的根节点。


  • 不管node是否插入成功,他都将返回当前二叉树的根节点


  • 以此来连接整个树结构。


// 插入元素
    public void add(E e) {
        root = add(root, e);
    }
    private Node add(Node node, E e) {
        if(node == null) {
            size++;
            return new Node(e);
        }
        if(e.compareTo(node.e) > 0) { // 加入的节点大于当前根节点
            node.right = add(node.right, e);
        }else if(e.compareTo(node.e) < 0){
            node.left = add(node.left, e);
        }
        return node; // 最后返回根节点
    }


网络异常,图片无法展示
|


非递归实现插入元素


// 插入元素非递归写法
public void addNotRecursion(E e) {
    if(root == null) { // 如果头结点为空
        root = new Node(e);
    }else { // 头结点不为空
        Node cur = root;
        while (cur != null) {
            if(e.compareTo(cur.e) > 0) { // 大于根节点
               if(cur.right == null) { // 插入元素
                   cur.right = new Node(e);
                   size++;
                   return;
               }
                cur = cur.right;
            }else if(e.compareTo(cur.e) < 0){
                if(cur.left == null) { // 插入元素
                    cur.left = new Node(e);
                    size++;
                    return;
                }
                cur = cur.left;
            }else {
                return;
            }
        }
    }
}


查询元素


思路和插入元素差不多。


// 查询元素
    public boolean contains(E e) {
        return contains(root, e);
    }
    private boolean contains(Node node, E e) {
        if(node == null) { // 如果树为空,那么返回false
            return false;
        }
        if(e.compareTo(node.e) == 0) { // 查到
            return true;
        }else if(e.compareTo(node.e) > 0) { // 查右边树
            return contains(node.right, e);
        }else { // 查左边的元素
            return contains(node.left, e);
        }
    }


递归实现遍历元素


我们需要理解递归的过程,当我们递归到最后时,会返回结果,然后释放当前函数上下文,然后继续向后执行,所以每个节点都会查找左中右三个部分。


前序遍历


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 前序遍历
    public void preOrder() {
        preOrder(root);
    }
    private void preOrder(Node node) {
        if(node == null) {
        // System.out.println("null");
            return ;
        }
        System.out.println(node.e);
        // 遍历该节点的左子树
        preOrder(node.left);
        // 遍历该节点的右子树
        preOrder(node.right);
    }


中序遍历


就是从小到大遍历


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 中序遍历
    public void inOrder() {
        inOrder(root);
    }
    private void inOrder(Node node) {
        if(node == null) {
            return ;
        }
        inOrder(node.left);
        System.out.println(node.e);
        inOrder(node.right);
    }


后序遍历


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 后序遍历
    public void postOrder() {
        postOrder(root);
    }
    private void postOrder(Node node) {
        if(node == null) {
            return ;
        }
        postOrder(node.left);
        postOrder(node.right);
        System.out.println(node.e);
    }


非递归实现遍历元素


通过来对元素进行遍历。


  • 将栈顶元素弹出后


  • 将该弹出元素的右孩子压入栈


  • 再将弹出元素的左孩子压入栈。


  • 在压入栈的过程中,孩子为空的情况,将不压入。循环条件是栈不能为空。


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 前序遍历非递归实现
    private void preOrderRn() {
        Stack<Node> stack = new Stack<>();
        stack.push(root);
        while(!stack.isEmpty()) {
            Node pop = stack.pop();
            System.out.println(pop.e);
            // 如果栈不为空,那么将叶子节点加入栈中
            // 先加入右节点
            if(pop.right != null)
            stack.push(pop.right);
            if(pop.left != null)
            stack.push(pop.left);
        }
    }


层次遍历, 广度优先遍历


通过队列实现,当我们出队的时候,将叶子结点加入到队列中。循环条件是队列不能为空


// 层次遍历,使用队列实现
public void levelOrder() {
    Queue<Node> queue = new LinkedList<>();
    queue.add(root);
    while (!queue.isEmpty()) { // 如果队列不为空
        // 出队
        Node cur = queue.remove();
        System.out.println(cur.e);
        if(cur.left != null) {
            queue.add(cur.left);
        }
        if(cur.right != null) {
            queue.add(cur.right);
        }
    }
}


网络异常,图片无法展示
|


删除二分搜索树的最小值


一直找树的left, 知道left为空,即是最小值。


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 查找最小值节点,指定根节点
    public Node findMin(Node root) {
        if(root == null) {
            return null;
        }
        Node cur = root;
        while (cur.left != null) {
            cur = cur.left;
        }
        return cur;
    }
    // 删除最小值
    public E removeMin() {
        // 删除的过程,将最小值的右子树添加到上一个根的左子树
        root = removeMin(root);
        return findMin(root).e;
    }
    private Node removeMin(Node node) {
        if(node.left == null) {
            Node currentItemRight = node.right;
            node.right = null;
            size--;
            // 返回当前元素的右子树
            return currentItemRight;
        }
        node.left = removeMin(node.left);
        return node;
    }


删除二分搜索树的最大值


一直找树的right, 直到right为空,即是最大值。


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 查找最大值节点,指定根节点
    public Node findMax(Node root) {
        if(root == null) {
            return null;
        }
        Node cur = root;
        while (cur.right != null) {
            cur = cur.right;
        }
        return cur;
    }
    // 删除最大值
    public E removeMax() {
        // 删除的过程,将最小值的右子树添加到上一个根的左子树
        root = removeMax(root);
        return findMax(root).e;
    }
    private Node removeMax(Node node) {
        if(node.right == null) {
            Node currentItemLeft = node.left;
            node.left = null;
            // 这里别忘记了size--
            size--;
            // 返回当前元素的左子树
            return currentItemLeft;
        }
        node.right = removeMax(node.right);
        return node;
    }


删任意节点 Hibbard Deletion


删除只有一个叶子节点的节点和上面的差不多,但是删除有左右叶子节点的节点就很麻烦了。


网络异常,图片无法展示
|


网络异常,图片无法展示
|


// 删除任意元素
    public void remove(E e) {
        root = remove(root, e);
    }
    private Node remove(Node node, E e) {
        if(node == null) {
            return null;
        }
        // 查看删除的元素是否有左右节点
        if(e.compareTo(node.e) < 0) {
            node.left = remove(node.left, e);
            return node;
        }else if(e.compareTo(node.e) > 0) {
            node.right = remove(node.right, e);
            return node;
        }else { // node.e = e;
            if(node.left == null) { // 当前节点左子树为空,可能右子树不为空
                Node rightNode = node.right;
                node.right = null;
                size --;
                return rightNode; // 返回删除节点的右子树
            }
            if(node.right == null) { // 当前节点的右子树为空,可能左子树不为空
                Node leftNode = node.left;
                node.left = null;
                size --;
                return leftNode; // 返回给上个节点,即上一个函数执行上下文,所以即是改删除节点的父节点。
            }
            // 处理该节点有左右子树
            // 找到比待删除节点大的最小节点,即待删除节点的右子树的最小节点。被称为后继节点
            // 然后用这个节点代替待删除节点的位置。
            Node successor = findMin(node.right);
            // 改变该后继节点的left指向
            successor.left = node.left;
            // 改变该后继节点的right指向。即删除了该后继节点的子树
            successor.right = removeMin(node.right);
            node.left = node.right = null;
            return successor;
        }
    }


对树的递归调用的总结


  • 注意将节点连接起来


  • 当当前值小于node,那么传入左子树。


  • 当当前值大于node, 那么传入右子树。


  • 找到终止递归的条件。node == null。


相关文章
|
2月前
|
算法
数据结构之博弈树搜索(深度优先搜索)
本文介绍了使用深度优先搜索(DFS)算法在二叉树中执行遍历及构建链表的过程。首先定义了二叉树节点`TreeNode`和链表节点`ListNode`的结构体。通过递归函数`dfs`实现了二叉树的深度优先遍历,按预序(根、左、右)输出节点值。接着,通过`buildLinkedList`函数根据DFS遍历的顺序构建了一个单链表,展示了如何将树结构转换为线性结构。最后,讨论了此算法的优点,如实现简单和内存效率高,同时也指出了潜在的内存管理问题,并分析了算法的时间复杂度。
65 0
|
6天前
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
42 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
6天前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
33 12
|
6天前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
32 10
|
6天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
28 2
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
81 5
|
2月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
119 16
|
2月前
|
算法
数据结构之文件系统模拟(树数据结构)
本文介绍了文件系统模拟及其核心概念,包括树状数据结构、节点结构、文件系统类和相关操作。通过构建虚拟环境,模拟文件的创建、删除、移动、搜索等操作,展示了文件系统的基本功能和性能。代码示例演示了这些操作的具体实现,包括文件和目录的创建、移动和删除。文章还讨论了该算法的优势和局限性,如灵活性高但节点移除效率低等问题。
73 0
|
3月前
|
存储 算法 关系型数据库
数据结构与算法学习二一:多路查找树、二叉树与B树、2-3树、B+树、B*树。(本章为了解基本知识即可,不做代码学习)
这篇文章主要介绍了多路查找树的基本概念,包括二叉树的局限性、多叉树的优化、B树及其变体(如2-3树、B+树、B*树)的特点和应用,旨在帮助读者理解这些数据结构在文件系统和数据库系统中的重要性和效率。
37 0
数据结构与算法学习二一:多路查找树、二叉树与B树、2-3树、B+树、B*树。(本章为了解基本知识即可,不做代码学习)
|
3月前
|
Java C++
【数据结构】探索红黑树的奥秘:自平衡原理图解及与二叉查找树的比较
本文深入解析红黑树的自平衡原理,介绍其五大原则,并通过图解和代码示例展示其内部机制。同时,对比红黑树与二叉查找树的性能差异,帮助读者更好地理解这两种数据结构的特点和应用场景。
48 0