码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • 笛卡尔树(Cartesian Tree)


    一、简介

            笛卡尔树是一种特定的二叉树数据结构,由数组存储。在范围最值、范围top k查询方面广泛应用。

            笛卡尔树的性质:

    1. 树中的元素满足二叉搜索树性质,要求按照中序遍历得到的序列为原数组序列
    2. 树中节点满足堆性质,节点的key值要大于其左右子节点的key值

            笛卡尔树中每个节点都有两个权值 (ki,wi) ,其中 ki 满足二叉搜索树, wi 满足二叉堆。

           当按照index从1到n(或者从0到n-1)的顺序将数组中的每个元素插入到笛卡尔树中时,当前要被插入的元素的index值最大,因此根据二叉搜索的性质需要沿着当前已经完成的笛卡尔树的根的右子树链搜索。 
           由于笛卡尔树要满足堆的性质(以最大堆为例),父节点的key值要大于子节点的key值,所以沿着树根的右子树链往下走,直到搜索到的节点的key值小于等于当前要插入节点的key值。 
           此时,便找到了当前结点需要插入的位置,记为P。此时P下方的节点的key值肯定小于当前被插入节点的key,但是index也小于当前插入节点的index(即需要在二叉搜索树中当前结点之前的位置),所以将当前节点插入到P的位置,同时将以P为根的子树挂载到新插入的节点的左子树

            图1讲解了笛卡尔树的存储方式。

    图1 - 笛卡尔树的存储方式示意图

    二、代码实现

    1. struct DTreeNode{
    2. int index;//在原来数组中的索引
    3. int key;//节点的key,即原数组中 Array[index]值
    4. DTreeNode* left;//左子节点
    5. DTreeNode* right;//右子节点
    6. DTreeNode* parent;//父节点
    7. DTreeNode(int i,int k):
    8. index(i),key(k),left(NULL),right(NULL),parent(NULL){};
    9. };
    10. DTreeNode* BuildTree(int* arr,int n)
    11. {
    12. stacknode_stack;
    13. DTreeNode* node=NULL,*new_node;
    14. for(int i=0; i
    15. {
    16. new_node=new DTreeNode(i,arr[i]);
    17. while(!node_stack.empty())
    18. {
    19. node=node_stack.top();
    20. if(node->key>new_node->key) //直到栈顶的元素的key大于当前结点的key
    21. {
    22. if(node->right)//将原来的右子链挂载到new_node的左子树
    23. {
    24. node->right->parent=new_node;
    25. new_node->left=node->right;
    26. }
    27. node->right=new_node;//将新插入的节点插入,作为右链的最后
    28. new_node->parent=node;
    29. break;
    30. }
    31. node_stack.pop();
    32. }
    33. node_stack.push(new_node);
    34. }
    35. //找出栈顶元素,就是笛卡尔树的根
    36. while(!node_stack.empty())
    37. {
    38. node=node_stack.top();
    39. node_stack.pop();
    40. }
    41. return node;
    42. }
    43. void InorderTravel(DTreeNode* root)//非递归进行中序遍历
    44. {
    45. stacknode_stack;
    46. DTreeNode* node=root;
    47. while(!node_stack.empty()||node)
    48. {
    49. if(node)
    50. {
    51. node_stack.push(node);
    52. node=node->left;
    53. }
    54. else
    55. {
    56. node=node_stack.top();
    57. node_stack.pop();
    58. cout<<"travel node, index = "<index<<", value = "<key<
    59. node=node->right;
    60. }
    61. }
    62. }

    参考文献:笛卡尔树 - 走看看笛卡尔树是一种同时满足二叉搜索树和堆的性质的数据结构。 可在一个数组上构造出来(时间复杂度可以达到O(n))。树中节点有几个属性, key(节点元素的大小)、index(节点在原数组中的索引)、lefhttp://t.zoukankan.com/gtarcoder-p-4702853.html 笛卡尔树 —— 二叉搜索树和堆的结合 - 知乎笛卡尔树是一种利用 单调栈创建的数据结构,用于维护区间最值与区间的关系。我们来回想下二叉搜索树的性质: 对于每个节点,满足左小右大。 对二叉搜索树进行中序遍历,可以获得排序后的序列。 二叉堆的性质: 上…https://zhuanlan.zhihu.com/p/493897833


    以上就是本文的全部内容啦!感谢阅读! 

  • 相关阅读:
    【图像生成Metrics】快速计算FID、KID、IS、PPL
    双网卡网络设置:有线网卡优先级高于无线网卡
    OpenGL进阶(一)之帧缓冲FrameBuffer
    网络安全竞赛C模块批量拿值脚本
    在linux下如何使用yum命令查看安装了哪些软件包
    CLIP扩展
    分布式之业务高可用
    Win10怎么设置待机时间
    hive orc文件读取出错
    nacos 注解
  • 原文地址:https://blog.csdn.net/weixin_46522531/article/details/126573883
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号