数据结构:构建完全二叉查找树

文章目录

    • 1、步骤 1: 对给定数组排序
    • 2、步骤 2: 递归构建完全二叉查找树
    • 3、注意
    • 4、在有序数组中寻找根结点位置
    • 5、代码实现
    • 6、其他方法?
      • 基本思路
      • 插入操作
      • 删除操作
      • 特别考虑

  对于一个给定序列的二叉查找树,有很多种,但是完全二叉查找树只有一种,因为树形是确定的,我们按照中根序列遍历的顺序也是按照数值大小顺序的,因此 按照树结点的中根遍历顺序 将 序列按照数值大小顺序 填入这颗树形确定的完全二叉树中,可以得到一颗唯一确定的完全二叉树。并且对于根结点,其左子树的所有结点都小于根结点,右子树的所有结点都大于等于根结点。
  构建一颗 完全二叉查找树(Complete Binary Search Tree,CBST)的算法关键在于如何保证二叉树的完全性质,同时维护二叉查找树(BST)的特性:对于任意节点,其左子树的所有节点的值小于该节点的值,其右子树的所有节点的值大于等于该节点的值。

以下是构建完全二叉查找树的一种有效算法:

1、步骤 1: 对给定数组排序

由于二叉查找树的中序遍历结果是一个有序数组,**首先要对输入的无序数组进行排序。**根结点的位置是可以唯一确定的。

2、步骤 2: 递归构建完全二叉查找树

  1. 选择中间元素作为树的根:从排序好的数组中选择中间的元素作为二叉查找树的根节点。如果有两个中间元素,选择它们中的任意一个。
  2. 递归构建左子树和右子树:将根节点左侧的元素递归地构建成左子树,将根节点右侧的元素递归地构建成右子树。

3、注意

  • 完全二叉树的定义是除了最后一层外,每一层都是完全填满的,而最后一层从左向右填充。上述算法构建的是一棵高度平衡的二叉查找树,但不一定满足完全二叉树的严格定义。构建满足完全二叉树定义的二叉查找树需要更复杂的逻辑来确保所有层(除了可能的最后一层)都完全填满。
  • 确保完全性的一个方法是通过计算给定元素数量所能构建的完全二叉树的最大高度,然后在构建过程中控制树的形态,使其满足完全二叉树的性质。

4、在有序数组中寻找根结点位置

在这里插入图片描述

完全二叉树的性质:

  • 性质的数学公式换算:
    • 树的结点个数n 取对数 可以得到树高: Treeheight=log2(n)
    • 通过树高 可以得到 除最后一层之外的右子树结点个数:rightTree1=pow(2,Treeheight-1)-1
    • 通过树高 以及 树的结点个数 可以得到树的最后一层结点个数:lastLevelNodes=n-(pow(2,Treeheight)-1)
    • 最后可以得到 右子树结点个数:rightTree=rightTree1+((lastLevelNodes-pow(2,Treeheight-1))>0?(lastLevelNodes-pow(2,Treeheight-1)) :0)
  • 直接求满了的层的总结点个数:
    • 利用快速幂思想,即取出总结点数n的二进制位数的最高位。
    • 然后计算即可得到右子树的个数

5、代码实现

  • 数学性质
#include<bits/stdc++.h>
using namespace std;

struct TreeNode {
    int val;
    TreeNode *left, *right;
    TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
};

// 构建完全二叉查找树
TreeNode* createTree(int start, int end, vector<int>& sortedValues) {
    if (start > end) return nullptr;

    // 计算完全二叉树的最后一层前的所有节点数和最后一层的节点数
    int totalNodes = end - start + 1;
    int fullLevels = log2(totalNodes + 1);
    int lastLevelNodes = totalNodes - (pow(2, fullLevels) - 1);
    int leftSubtreeNodes = pow(2, fullLevels - 1) - 1 + min((int)pow(2, fullLevels - 1), lastLevelNodes);

    // 根据左子树的节点数确定根节点
    int rootIndex = start + leftSubtreeNodes;
    TreeNode* root = new TreeNode(sortedValues[rootIndex]);
    root->left = createTree(start, rootIndex - 1, sortedValues);
    root->right = createTree(rootIndex + 1, end, sortedValues);

    return root;
}

// 层次遍历
void traversal(TreeNode* root) {
    if (!root) return;
    queue<TreeNode*> q;
    q.push(root);
    while (!q.empty()) {
        TreeNode* current = q.front(); q.pop();
        cout << current->val << " ";
        if (current->left) q.push(current->left);
        if (current->right) q.push(current->right);
    }
}

int main() {
    int n;
    cin >> n;
    vector<int> values(n);
    for (int& value : values) {
        cin >> value;
    }
    sort(values.begin(), values.end());

    TreeNode* root = createTree(0, n - 1, values);
    traversal(root);
    return 0;
}
  • 直接计算
#include<bits/stdc++.h>
using namespace std;
struct TreeNode{
    TreeNode * left=nullptr;
    TreeNode * right=nullptr;
    int val;
    TreeNode (int v=0):val(v) {}
};
TreeNode * createTree(int left,int right,vector<int> & baby){//O(nlogn)
    if(left>right) return nullptr;
    if(left==right) return new TreeNode(baby[left]);
    TreeNode * root=nullptr;
    int flag=right-left+2;
    int b=1;
    while(flag!=1){//O(logn) 每次建树logn,n个结点共nlogn
        b*=2;
        flag>>=1;
    }
    int rightTree_size=b/2-1;
    if(rightTree_size+2-b-b/2>0){
       rightTree_size+=rightTree_size+2-b-b/2;
    }
    root=new TreeNode(baby[right-rightTree_size]);
    root->left=createTree(left,right-rightTree_size-1,baby);
    root->right=createTree(right-rightTree_size+1,right,baby);
    return root;
}
int main(void){
    ios_base::sync_with_stdio(false);
    cin.tie(0);
    int n;
    cin>>n;
    vector<int> baby;
    for(int i=0;i<n;++i){
        int j;cin>>j;
        baby.emplace_back(j);
    }
    sort(baby.begin(),baby.end());//O(nlogn)
    createTree(0,n-1,baby);
    return 0;
}

6、其他方法?

动态构建完全二叉查找树(CBST)比静态构建更为复杂,因为需要在插入和删除操作发生时持续保持树的完全二叉树性质。一种方法是在每次插入或删除后重建树,但这显然效率不高。另一种思路是通过维护额外的信息和采用特定的插入/删除策略来尽可能保持树的完全性。下面是一个较为高效的动态构建完全二叉查找树的策略:

基本思路

动态维护一个完全二叉查找树,主要挑战在于插入和删除操作。对于插入操作,需要找到树中最后一个节点并在适当的位置插入新节点以保持完全二叉树的性质。对于删除操作,则需要找到可以替换被删除节点的节点(通常是树中的最后一个节点)并调整树以保持其完全性质。

插入操作

  1. 计算完全二叉树的深度:基于树当前的节点数量,可以计算出树的深度。
  2. 定位插入点:根据完全二叉树的性质,新节点应当被插入为最后一个节点。可以通过深度和节点数定位到这个位置。
  3. 执行插入:在定位到的位置插入新节点。如果需要,调整树的结构以保持二叉查找树的性质。

删除操作

  1. 定位并删除目标节点:找到需要删除的节点,将其删除。如果这个节点不是最后一个节点,则需要找到最后一个节点。
  2. 用最后一个节点替换删除的节点(如果被删除的节点不是最后一个节点):将最后一个节点移动到被删除节点的位置。
  3. 调整树:可能需要对树进行调整,以保持二叉查找树的性质。

特别考虑

  • 这种方法要求你能快速定位到树的最后一个节点,这可能需要维护额外的信息,如每层的节点数。
  • 插入和删除操作可能导致需要重新平衡树,以保持二叉查找树的性质。
  • 动态维护完全二叉树的复杂度主要在于定位最后一个节点和维护二叉查找树的性质。

动态地维护一个完全二叉查找树在实践中较为罕见,因为它要求在每次操作后都严格保持完全二叉树的性质,这可能导致较高的复杂性和成本。通常,人们会使用其他类型的自平衡二叉搜索树(如AVL树、红黑树)来获得较好的平均性能,尽管这些树不保证完全性质。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Windows安装Kibana

下载 注意&#xff1a;为了避免一些稀奇古怪的问题&#xff0c;kibana版本最好和es版本保持一致。 es版本查看&#xff1a; 官网下载地址&#xff1a; Download Kibana Free | Get Started Now | Elastichttps://www.elastic.co/cn/downloads/kibana如果是下载最新的&#x…

41---音频电路设计

视频链接 音频电路设计01_哔哩哔哩_bilibili 音频电路设计 1、音频基本介绍 1.1、设备 1.1.1、音频接口 型号&#xff1a;ABA-JAK-038-K44 电脑主机上的音频输出插口&#xff0c;一个是粉色的&#xff0c;用来连接麦克风或话筒&#xff0c;一个是绿色的&#xff0c;用来连…

【数据结构与算法】:归并排序和计数排序

1. 归并排序 归并排序是一种效率仅次于快速排序的排序算法。它有非递归和递归两种实现方式(本文只讲述递归实现&#xff0c;非递归实现以后有专门的文章)。 其实&#xff0c;归并排序也叫外排序。它不仅可以对内存中的数据进行排序&#xff0c;还能对文件里的数据排序。 比如&…

网站压力测试和Locust

一、压力测试介绍 网站压力测试是一种评估网站性能、可靠性和稳定性的方法。它通过模拟大量用户同时访问网站,来测试网站的响应时间、吞吐量、资源利用率等指标,从而发现网站的潜在问题和瓶颈。下面我将从几个方面详细介绍网站压力测试: 1、压力测试的目的 评估网站在高并发…

路由器端口映射是什么意思?

路由器端口映射是一种网络配置技术&#xff0c;在私有网络中允许外部网络访问特定的服务或应用程序。通过将路由器的端口映射到内部客户端设备&#xff0c;可以实现从公共网络访问内部网络资源的目的。 天联组网介绍 天联是一款异地组网内网穿透产品&#xff0c;由北京金万维科…

【Qt】:常用控件(九:容器类控件)

常用控件 一.Group Box&#xff08;分组框&#xff09;二.Tab Widget&#xff08;标签页&#xff09; 一.Group Box&#xff08;分组框&#xff09; 使用QGroupBox实现一个带有标题的分组框.可以把其他的控件放到里面作为一组.这样看起来能更好看一点.&#xff08;换言之&…

复现bytetrack时,安装依赖项报错“: ERROR: Failed building wheel for lap

报错原因&#xff1a; lap 库的构建失败&#xff0c;因为缺少了 NumPy 库。 解决办法&#xff1a; 安装 NumPy 库&#xff1a;NumPy 是 Python 中用于科学计算的基础库&#xff0c;lap 依赖于它 pip install numpy 重新安装 lap 库&#xff1a; pip install lap

代码随想录|Day32|贪心算法 part02|● 122.买卖股票的最佳时机II ● 55. 跳跃游戏 ● 45.跳跃游戏II

122.买卖股票的最佳时机II class Solution: def maxProfit(self, prices: List[int]) -> int: result 0 for i in range(len(prices) - 1): count prices[i1] - prices[i] if count > 0: result count return result 方法二&#xff1a;把if条件变成max class Solutio…

智能配电能效平台与照明系统在某地下污水处理厂中的应用

安科瑞薛瑶瑶18701709087 1、引言 随着互联网、芯片技术、通信传输的技术革新和成熟&#xff0c;智能照明已经广泛应用于居民生活和工业发展领域。传统的工业照明设计&#xff0c;常在门口附近设置集中控制箱&#xff0c;由控制箱内相应开关控制照明。当工厂面积较大&#xf…

ONERugged车载平板终端:提升港口运输水平

现代港口是国际贸易中至关重要的枢纽&#xff0c;而提高港口运输效率对于促进贸易流通和经济发展至关重要。近年来&#xff0c;车载平板技术的快速发展为港口运输行业带来了巨大的变革和机遇。车载平板的广泛应用不仅提高了港口的操作效率&#xff0c;还改善了货物跟踪、通信和…

Vue3中使用的富文本编辑器(详细实现流程)

文章目录 1. 前言2. 项目初始化3. 下载4. 使用富文本编辑器5. 注意点6. 效果图 1. 前言 有不少的前端需求都需要使用到富文本编辑器,但是富文本编辑器百花齐放,每次使用可能都会重新找一个编辑器,所以有了这篇文章. 当项目中需要使用到富文本编辑器时,可以直接按照这篇文章的步…

动态分区算法

一个不知名大学生&#xff0c;江湖人称菜狗 original author: Jacky Li Email : 3435673055qq.com Time of completion&#xff1a;2024.04.09 Last edited: 2024.04.09 目录 动态分区算法 第1关&#xff1a;首次适应算法 任务描述 相关知识 内存分配 内存回收 编程要求…

chronyd服务

一、介绍 chronyd服务是CentOS8系统之后提供时间服务的应用&#xff0c;和之前的ntp服务功能是一样的。 chronyd服务的配置文件默认存在在/etc/chrony.conf中。 chronyd服务的开启方式和关闭&#xff1a; systemctl start chronyd systemctl status chronyd systemctl st…

每天好好学习java第一天--复习巩固基础

1.浮点数数据特殊&#xff1a; float z 2.0e8F; float类型要在后面加f或者F。但是double类型可以省略。 2.强制转换数据类型&#xff1a; 格式&#xff1a; (类型名)变量名 例 float z 2.0f; int x(int)z; 3.逻辑运算符 注意异或 4.条件运算符 每天学习一会java&…

性能分析-数据库与磁盘知识

数据库 数据库&#xff0c;其实是数据库管理系统dbms。 数据库管理系统&#xff0c; 常见&#xff1a; 关系型数据库&#xff1a; mysql、pg、 库的表&#xff0c;表与表之间有关联关系&#xff1b; 表二维表统一标准的SQL&#xff08;不局限于CRUD&#xff09;非关系型数据…

配置VM开机自启动

1. 在此电脑-右键选择“管理”-服务和应用程序-服务中找到VMware Workstation Server服务&#xff08;新版名称也可能是VMware自启动服务&#xff0c;自己找一下&#xff0c;服务属性里有描述信息的&#xff09;&#xff0c;将其启用并选择开机自动启动 新版参考官方文档&…

C语言 函数——函数原型

目录 如何合并成一个完整的程序&#xff1f; 函数原型与函数定义的区别 函数原型的作用 如何合并成一个完整的程序&#xff1f; 问题&#xff1a;在一个函数中调用另一个函数&#xff0c;需要具备哪些条件呢&#xff1f; 若函数的定义出现在函数调用之前 若函数的定义出现…

跨云迁移实操:AWS RDS for mysql 迁移至腾讯云mysql --DTS方式

实操场景&#xff1a;从AWS RDS for mysql 迁移至腾讯云云数据库Mysql&#xff0c;通过腾讯云数据传输服务DTS,进行实时全量增量迁移. 下面九河云给大家带来具体实践介绍 购买迁移数据库--目的端机器&#xff08;腾讯云MYSQL&#xff09; 可以源端为5.7所以新建一个参数模版 其…

4 万字 102 道Java经典面试题总结(2024修订版)- 多线程篇

&#x1f345; 作者简介&#xff1a;哪吒&#xff0c;CSDN2021博客之星亚军&#x1f3c6;、新星计划导师✌、博客专家&#x1f4aa; &#x1f345; 哪吒多年工作总结&#xff1a;Java学习路线总结&#xff0c;搬砖工逆袭Java架构师 &#x1f345; 技术交流&#xff1a;定期更新…

全新AI天空任意生成解决方案,颠覆传统换天效果

在数字化时代&#xff0c;影像创作已经成为企业展示品牌形象、传递信息的重要手段。特别是在汽车拍摄和旅行拍摄等场景中&#xff0c;天空作为画面中不可或缺的元素&#xff0c;其表现往往直接关系到作品的质感和吸引力。然而&#xff0c;传统的天空替换技术往往操作繁琐、效果…