Algorithm-Pattern
  • Introduction
  • C & C++
    • C语言
      • C/C++编译器
      • 宏的使用
      • 编译过程
      • 指针 & 数组
      • 柔性数组
      • 函数指针 & 回调函数
      • C标准库之stdio
      • C标准库之string
      • C标准库之errno
      • C标准库之stdarg
      • C标准库之regex
    • C++基础语法
      • 自增(++) & 自减(--)
      • c语言到c++
      • 可变模板参数
      • 强制类型转换
      • C/C++类型转换的本质
      • 指针 & 引用
      • const的用法
      • static的用法
      • 重要的关键字(一)
      • 重要的关键字(二)
      • 内存申请和释放
      • 内联函数
      • 函数 & 运算符重载
      • 面向对象之封装
      • 构造函数 & 析构函数
      • 面向对象之继承
      • 面向对象之多态
      • 泛型编程
      • 异常
      • 再谈函数指针
    • C++并发编程
      • C++的锁
      • 并发与多线程
    • C++高级特性
      • 函数对象
      • 移动语义 & 完美转发
      • lambda表达式
      • RTTI技术
      • RAII技术
      • 智能指针
      • 模板的特化
      • C++静态库和动态库
      • 内存溢出和内存泄漏
    • STL基础
      • String
      • array/vector/list
      • deque/priority_queue
      • set/map
      • unordered_set/unordered_map
      • algorithm_1
      • functional
      • allocator
    • C++标准库
      • IO
      • Tuple
      • regex
      • bitset
      • numeric
    • STL深入源码
      • vector内部实现
      • deque内部实现
      • sort函数实现
    • 第三方库
      • JsonCpp
      • ProtoBuf
  • 数据结构
    • 线性表
    • 字符串
    • 栈和队列
    • 二叉树
    • 平衡二叉树
    • 平衡多路搜索树
    • 树结构的延申
    • 图
    • 二进制
    • 散列表
  • 算法基础
    • 排序算法
    • 查找算法
    • 数学问题
    • 并查集
    • 递归算法
    • 附加——主定理
    • Catalan数
  • 算法设计思想
    • 滑动窗口思想
    • BFS/DFS
    • 二分法
    • 回溯法
    • 贪心算法
    • 分治法
    • 动态规划
    • 分支限界算法
    • 有限状态机(FSM)
  • LeetCode系列
    • 死磕二叉树
    • 股票买卖问题
    • 打家劫舍问题
    • 跳跃游戏问题
    • 括号匹配问题
    • 石子游戏问题
    • 子序列问题
    • 数组 & 矩阵
    • 排列 & 组合
  • 经典算法问题
    • 几何问题
    • 区间问题
    • 背包问题
    • 石子堆问题
    • 表达式求值
  • 面试题
    • 数据结构和算法基础
    • 程序设计题
      • 实现双线程交替打印
      • C++实现读写锁
      • 实现阻塞队列
      • 实现环形队列
      • 实现线程池
      • 实现智能指针
      • 实现string类
      • 实现高性能local cache
      • 实现内存池
      • 生产者-消费者模型
      • 设计定时器
    • 经典的算法题
    • C++面试题总结
    • 面试算法题总结
由 GitBook 提供支持
在本页
  • 一、二叉搜索树
  • 1、定义
  • 2、验证二叉搜索树
  • 3、应用
  • 二、平衡二叉(搜索)树
  • 1、AVL树
  • 2、红黑树

这有帮助吗?

  1. 数据结构

平衡二叉树

二叉搜索树和平衡二叉搜索树的相关概念和算法实现。

上一页二叉树下一页平衡多路搜索树

最后更新于4年前

这有帮助吗?

一、二叉搜索树

1、定义

  • 每个节点中的值必须大于(或等于)存储在其左侧子树中的任何值。

  • 每个节点中的值必须小于(或等于)存储在其右子树中的任何值。

2、验证二叉搜索树

给定一个二叉树,判断其是否是一个有效的二叉搜索树。

思路 1:中序遍历,检查结果列表是否已经有序

思路 2:分治法,判断 左 MAX < 根 < 右 MIN,如果该二叉树的左子树不为空,则左子树上所有节点的值均小于它的根节点的值; 若它的右子树不空,则右子树上所有节点的值均大于它的根节点的值;它的左右子树也为二叉搜索树。

bool isValidBST(TreeNode* root){
    if(root && !root->left && !root->right)
        return true;
    stack<TreeNode *> istack;
    TreeNode *ptr = root;
    long long last_val = (long long)INT_MIN - 1;  // 扩大int的范围,考虑边界值
    while(ptr || !istack.empty()){
        while(ptr) {
            istack.push(ptr);
            ptr = ptr->left;
        }
        TreeNode *curr_ptr = istack.top();
        if(curr_ptr->val <= last_val)
            return false;
        last_val = curr_ptr->val;
        istack.pop();
        ptr = curr_ptr->right;
    }
    return true;
}
bool helper(TreeNode* root, long long lower, long long upper) {
    if(root == NULL)
        return true;
    if(root->val <= lower || root->val >= upper)
        return false;
    return helper(root->left, lower, root->val) 
                  && helper(root->right, root->val, upper);
}

bool isValidBST(TreeNode* root){
    return helper(root, LONG_MIN, LONG_MAX);
}

给定一个二叉树,判断它是否是高度平衡的二叉树。

一棵高度平衡二叉树定义为:一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过1。

方法一:自顶向下的递归

if (root == NULL)
    return true;
if (abs(height(root.left) - height(root.right)) > 1)
    return false;
else
    return isBalanced(root.left) && isBalanced(root.right);

方法二:自底向上的递归

方法一计算 height 存在大量冗余。每次调用 height 时,要同时计算其子树高度。但是自底向上计算,每个子树的高度只会计算一次。可以递归先计算当前节点的子节点高度,然后再通过子节点高度判断当前节点是否平衡,从而消除冗余。

bool isBalancedTree(TreeNode *root, int &subHeight){
    if(!root){
        subHeight = 0;
        return true;
    }
    int x,y;
    bool bLeft = isBalancedTree(root->left, x);
    bool bRight = isBalancedTree(root->right, y);
    bool bIt = std::abs(x - y) > 1 ? false : true;
    subHeight = x > y ? x + 1 : y + 1;
    return bLeft && bRight && bIt;
}
bool isBalanced(TreeNode* root) {
    int height = 0;
    return isBalancedTree(root, height);
}

3、应用

给定二叉搜索树(BST)的根节点和要插入树中的值,将值插入二叉搜索树。 返回插入后二叉搜索树的根节点。 保证原始二叉搜索树中不存在新值。

给定一个二叉搜索树的根节点 root 和一个值 key,删除二叉搜索树中的 key 对应的节点,并保证二叉搜索树的性质不变。返回二叉搜索树(有可能被更新)的根节点的引用。

平衡二叉树(Balanced Binary Tree)又被称为AVL树。它具有以下性质:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。

R-B Tree,全称是Red-Black Tree,又称为“红黑树”,它一种特殊的二叉查找树(近似平衡的二路查找树)。红黑树的每个节点上都有存储位表示节点的颜色,可以是红(Red)或黑(Black)。

  1. 每个节点或者是黑色,或者是红色。

  2. 根节点是黑色。

  3. 每个叶子节点(NIL)是黑色。 [注意:这里叶子节点,是指为空(NIL或NULL)的叶子节点!]

  4. 如果一个节点是红色的,则它的子节点必须是黑色的。

  5. 从一个节点到该节点的子孙节点的所有路径上包含相同数目的黑节点。确保没有一条路径会比其他路径长出俩倍。因而,红黑树是相对是接近平衡的二叉树。

从性质5又可以推出:

  • 性质5.1:如果一个结点存在黑子结点,那么该结点肯定有两个子结点

根据定义,一棵二叉树 TTT 存在节点 p∈Tp\in Tp∈T ,满足 ∣height(p.left)−height(p.right)∣>1∣height(p.left)−height(p.right)∣>1∣height(p.left)−height(p.right)∣>1 时,它是不平衡的。下图中每个节点的高度都被标记出来,高亮区域是一棵不平衡子树。

定义方法 heightheightheight ,用于计算任意一个节点 p∈Tp\in Tp∈T 的高度:

height(p)={−1,p is an empty subtree i.e.null1+max(height(p.left),height(p.right)),otherwiseheight(p)=\begin{cases} -1, & p\ \text{is an empty subtree i.e.} null \\ 1+max(height(p.left),height(p.right)), & \text{otherwise} \end{cases}height(p)={−1,1+max(height(p.left),height(p.right)),​p is an empty subtree i.e.nullotherwise​

接下来就是比较每个节点左右子树的高度。在一棵以 rrr 为根节点的树 TTT 中,只有每个节点左右子树高度差不大于 1 时,该树才是平衡的。因此可以比较每个节点左右两棵子树的高度差,然后向上递归。

二、平衡二叉(搜索)树

1、AVL树

一颗高度为n的平衡二叉树,需要的最少节点数为: F(n)=F(n−1)+F(n−2)+1,F(1)=1,F(2)=2F(n) = F(n - 1) + F (n - 2) + 1,F(1) = 1,F(2) = 2F(n)=F(n−1)+F(n−2)+1,F(1)=1,F(2)=2 。

2、红黑树

2.1、红黑树的特性:

定理:一棵含有n个节点的红黑树的高度至多为 2log2(n+1)2log_2(n+1)2log2​(n+1) 。

2.2、散列表、红黑树和跳表

散列表:插入删除查找都是 O(1)O(1)O(1) ,是最常用的,但其缺点是不能顺序遍历,扩容缩容的性能损耗很大。适用于那些不需要顺序遍历,数据更新不那么频繁的场景。

跳表(Skip list):插入删除查找都是 O(logn)O(logn)O(logn) ,能顺序遍历,实现简单。缺点是空间复杂度为 O(n)O(n)O(n) (磁盘IO相对比较频繁,因为要加载多级索引即重复多层节点),其顺序遍历和区间查找非常方便。

红黑树:插入删除查找都是 O(logn)O(logn)O(logn) ,中序遍历即是顺序遍历,比较稳定。缺点是难以实现,区间查找不方便。

✏️
🖋️
🖋️
💎
💎
✏️
validate-binary-search-tree
balanced-binary-tree
insert-into-a-binary-search-tree
delete-node-in-a-bst