笛卡尔树
来源:互联网 发布:linux打开elf文件 编辑:程序博客网 时间:2024/04/29 03:13
笛卡尔树简单介绍
笛卡尔树又称笛卡儿树,在数据结构中属于二叉树的一种。可以这么说:笛卡尔树是一棵二叉树,树的每个节点有两个值,一个为key,一个为value。光看key的话,笛卡尔树是一棵二叉搜索树,每个节点的左子树的key都比它小,右子树都比它大;光看value的话,笛卡尔树有点类似堆,根节点的value是最小(或者最大)的,每个节点的value都比它的子树要小(或者大)。笛卡尔树定义
无相同元素的数列构造出的笛卡尔树具有下列性质:1、结点一一对应于数列元素。即数列中的每个元素都对应于树中某个唯一结点,树结点也对应于数列中的某个唯一元素2、中序遍历(in-order traverse)笛卡尔树即可得到原数列。即任意树结点的左子树结点所对应的数列元素下标比该结点所对应元素的下标小,右子树结点所对应数列元素下标比该结点所对应元素下标大。3、树结构存在堆序性质,即任意树结点所对应数值大(或小)于其左、右子树内任意结点对应数值(即根节点为其子树的最值)根据堆序性质,笛卡尔树根结点为数列中的最大/小值,树本身也可以通过这一性质递归地定义:根结点为序列的最大/小值,左、右子树则对应于左右两个子序列,其结点同样为两个子序列的最大/小值。因此,上述三条性质唯一地定义了笛卡尔树。若数列中存在重复值,则可用其它排序原则为数列中相同元素排定序列,例如以下标较小的数为较小,便能为含重复值的数列构造笛卡尔树。笛卡尔树的实现
O(N^2)算法实现
①排序之后直接构造笛卡尔树的方法:首先将节点序列按照key从小到大排序,然后按照顺序插入节点,注意到排序之后,插入的节点的key值一定是树中最大的,所以只需查找最右端的路径,找到一个节点A[i]的value大于待插入节点的value,同时A[i]->right的value小于待插入节点的value。找到之后,只需将A[i]的right指向待插入的节点,A[i]的right原来指向的节点赋值给待插入节点的left指针。注意到查找最右路径的方向,如果从下到上查找,复杂度比较容易分析O(N)(因为查找过的节点必然会旋转到某个节点的左子节点,因此每个查找过的节点只会被查找一次),如果从上倒下,比较复杂(和最右端的最终的路径长度有关吧),会超过N,甚至更高,可能为O(N^2)。②利用排序加左旋的方法:就是一样先排序,然后使用treap插入节点,可以发现,所有的旋转都为左旋。这种方法也TLE了,这种方法有一个很重要的意义,就是分析了上个方法中从上到下扫描的复杂度。因为这两种方法的效率是等价的,都TLE。O(N)算法实现
我们将要将A的元素依次插入笛卡尔树C。每次插入都可能使树的形态发生变化。为了在O(N)的时间内完成整个插入过程,考虑C的右链,即根结点、根结点的右儿子、根结点的右儿子的右儿子……组成的链。注意这些元素的下标和值都是递增的。下标最大,即将要插入的元素A[i]一定是新树右链的最后一个元素。原来的右链中,值比A[i]大的元素在新树中不再属于右链,这些元素组成的链成为A[i]的左子树的右链;原来右链中的其它元素加上A[i]组成了新的右链。初看起来,寻找分界点的最佳方法是O(logN)时间的二分查找;但是对于整个过程来说,O(NlogN)的时间复杂度不是最优的。关键在于一旦一个元素比A[i]大,它就从右链中被永久地移除了。如果按照从后到前的顺序判断一个元素是否大于A[i],则每次插入的时间复杂度为O(k+1),k为本次插入中移除的右链元素个数。因为每个元素最多进出右链各一次,所以整个过程的时间复杂度为O(N)。用一个栈结构维护右链元素的下标,上述过程可以很容易地实现。(见下面代码部分)
笛卡尔树
0 0
- 笛卡尔树
- 笛卡尔树
- 笛卡尔树
- 笛卡尔树
- 笛卡尔树
- 笛卡尔树
- 笛卡尔树
- 笛卡尔树
- poj 2201 笛卡尔树
- sgu155 笛卡尔树
- 笛卡尔树
- poj2796Feel Good(笛卡尔树)
- PAT4-09. 笛卡尔树
- poj2051Fence Repair笛卡尔树
- 笛卡尔树cartesian tree
- 笛卡尔树构建
- pat-笛卡尔树
- poj 2201 笛卡尔树
- 2016.01.02JS滚轮事件详解
- linux内核学习(一)------------模块
- 周总结
- Python------异常处理
- 详解UICollectionView
- 笛卡尔树
- 工厂三兄弟之工厂方法模式(二):工厂方法模式概述
- wampserver安装过程中图标黄色的处理方法
- 树形节点的递归遍历
- Linux学习之Led驱动程序理解
- 在VirtualBox 5.0.12 for Linux版本上安装CentOS 6.6 x86_64系统
- 自定义语言的实现——解释器模式
- 怎样改变 unbutton 的title 和image 的位子
- 关于binary diff / patch 差分补丁工具的那些事