前言

最近做题目,已经不止一次用到笛卡尔树了。这种数据结构极为优秀,但是构造的细节很容易出错。因此写一篇文章做一个总结。

笛卡尔树 Cartesian Tree

引入问题

有N条的长条状的矩形,宽度都为1,第i条高度为Hi,相邻的竖立在x轴上,求最大的子矩形面积。

约定

1 ≤ N ≤ 105

1 ≤ Hi ≤ 109

分析

我们只需要求出每条矩形最多可以向两边拓展的宽度,就可以算出以这个矩形高度为高的最大子矩形面积。最后我们求一个最大值即可。

下面我们还是回到之前的笛卡尔树。

概念

笛卡尔树的树根是这一子树中键值最小(或最大)的元素;且对于某个序列的笛卡尔树,其任一子树的中序遍历恰好是对应了原序列中的一段连续区间。

性质

我们会发现笛卡尔树同时满足二叉树搜索和堆的性质:

  1. 中序遍历即为原数组序列
  2. 父节点的键值均小于(或大于)其左右子节点的键值

构造

我们可以利用单调栈在线性时间内对给定的数组序列构造出其笛卡尔树。

首先,由于笛卡尔树的中序遍历为原数组序列,那么我们设

Ti为当前序列中[1..i]区间的笛卡尔树,

那么,一定有:

第(i+1)个节点属于Ti最右边的那一条路径。

那么对于有已经构造好的Ti和第(i+1)个节点,我们只需要沿着Ti最右边的路径从下往上找,直到发现当前节点可以放的位置。

我们可以用一个单调栈来维护最右边的这条路径,当前根节点压在栈底,对于当前节点,我们将它与栈顶比较,若栈顶不能做它的父节点,则将栈顶弹出。

由于每个点进栈和出栈至多一次,因此这个构造的复杂度为O(n)的。

参考代码

节点用结构体定义

(这里我们用指针会方便许多)

struct Node {
int index, val; // index表示原数组的索引,val为当前节点的键值
Node *parent, *lefts, *rights; // 这三个指针分别指向父节点,左子节点,右子节点
Node(int id = , int v = , Node *l = NULL, Node *r = NULL) { index = id; val = v; lefts = l; rights = r; }
};

树构造函数

(这里要注意的细节较多)

Node * build(int arr[], int size) {    // 这里构建一个根节点为最小值的笛卡尔树
std::stack<Node * > S; // 存储最右边路径的栈
Node *now, *next, *last;
for (int i = ; i < size; i++) {
next = new Node(i, arr[i]); last = NULL; // last用来指向最后被弹出栈的元素(若有弹出),它的作用后面会写到
while (!S.empty()) {
if (S.top()->val < next->val) { // 若栈顶节点的键值比当前节点键值小了,那么当前节点就做栈顶节点的右子节点
now = S.top();
if (now->rights) { // 而栈顶节点的原右子节点要变成当前节点的左子节点(由于前面一定与当前节点比较过了,栈顶节点右子树的键值一定都比当前节点大)
now->rights->parent = next;
next->lefts = now->rights;
}
now->rights = next;
next->parent = now;
break;
}
last = S.top();
S.pop();
}
if (S.empty() && last) { // 这里为了特判一种可能出现的情况,就是当前节点把栈全部弹空了,就要把原先的根节点作为当前节点的左子节点
next->lefts = last;
last->parent = next;
}
S.push(next);
}
while (!S.empty()) now = S.top(), S.pop();
return now;
}

笛卡尔树Cartesian Tree的更多相关文章

  1. [模板] 笛卡尔树 && RMQ

    话说我noip之前为什么要学这种东西... 简介 笛卡尔树(Cartesian Tree) 是一种二叉树, 且同时具有以下两种性质: 父亲节点的值大于/小于子节点的值; 中序遍历的结果为原序列. 笛卡 ...

  2. POJ 2201 Cartesian Tree ——笛卡尔树

    [题目分析] 构造一颗笛卡尔树,然后输出这棵树即可. 首先进行排序,然后用一个栈维护最右的树的节点信息,插入的时候按照第二关键字去找,找到之后插入,下面的树成为它的左子树即可. 然后插入分三种情况讨论 ...

  3. NOIP2011pj表达式的值[树形DP 笛卡尔树 | 栈 表达式解析]

    题目描述 对于1 位二进制变量定义两种运算: 运算的优先级是: 先计算括号内的,再计算括号外的. “× ”运算优先于“⊕”运算,即计算表达式时,先计算× 运算,再计算⊕运算.例如:计算表达式A⊕B × ...

  4. 平衡树及笛卡尔树讲解(旋转treap,非旋转treap,splay,替罪羊树及可持久化)

    在刷了许多道平衡树的题之后,对平衡树有了较为深入的理解,在这里和大家分享一下,希望对大家学习平衡树能有帮助. 平衡树有好多种,比如treap,splay,红黑树,STL中的set.在这里只介绍几种常用 ...

  5. [乱搞]hdu 6406 Taotao picks apples 笛卡尔树+倍增

    题目链接 Problem Description There is an apple tree in front of Taotao's house. When autumn comes, n app ...

  6. Codeforces Round #114 (Div. 1) D. Wizards and Roads 笛卡尔树+树贪心+阅读题

    D. Wizards and Roads 题目连接: http://www.codeforces.com/contest/167/problem/D Description In some count ...

  7. 笛卡尔树 POJ ——1785 Binary Search Heap Construction

    相应POJ 题目:点击打开链接 Binary Search Heap Construction Time Limit: 2000MS   Memory Limit: 30000K Total Subm ...

  8. HDU - 1506 Largest Rectangle in a Histogram (单调栈/笛卡尔树)

    题意:求一个直方图中最大矩形的面积. 很经典的一道问题了吧,可以用单调栈分别求出每个柱子左右两边第一个比它低的柱子(也就相当于求出了和它相连的最后一个比它高的柱子),确定每个柱子的左右边界,每个柱子的 ...

  9. POJ-1785-Binary Search Heap Construction(笛卡尔树)

    Description Read the statement of problem G for the definitions concerning trees. In the following w ...

随机推荐

  1. 详细介绍redis的集群功能,带你了解真正意义上的分布式

    Redis 集群是一个分布式(distributed).容错(fault-tolerant)的 Redis 实现, 集群可以使用的功能是普通单机 Redis 所能使用的功能的一个子集(subset). ...

  2. 云计算时代,传统企业 IT 从业者如何做好转型?

    本文来源于国外社区 DZone,作者 Dennis O'Reilly 撰写过多篇关于云计算.混合云等内容的文章,本文内容围绕云计算时代,企业纷纷上云,传统 IT 从业者如何做好转型. 本文由“数梦工场 ...

  3. 在NodeJS中使用Redis缓存数据

    Redis数据库采用极简的设计思想,最新版的源码包还不到2Mb.其在使用上也有别于一般的数据库. node_redis redis驱动程序多使用 node_redis 此模块可搭载官方的 hiredi ...

  4. Team Member Introduction and Division of Work

    Team leader Name:宋天舒 Student Number:12061166 Interested In: Information safety. Responsible For: Des ...

  5. 实验1 熟悉Linux开发环境 实验报告

    参见http://www.cnblogs.com/lhc-java/p/4970269.html

  6. 利用Hibernate子查询(in) 得到部分字段(实体类的构造函数)

    感人= = 终于弄好了 String hql="select new Shop(s.strid,s.shopname,s.tradearea,s.discountinfo,s.beginti ...

  7. 【Coursera】应用机器学习的建议

    偏差方差权衡 使用较小的神经网络,类似于参数较少的情况,容易导致高偏差和欠拟合,但计算代价较小使用较大的神经网络,类似于参数较多的情况,容易导致高方差和过拟合,虽然计算代价比较大,但是可以通过归一化手 ...

  8. 树莓派与Arduino Leonardo使用NRF24L01无线模块通信之基于RF24库 (二) 发送自定义数据

    在我的项目里,树莓派主要作为中心节点,用于接收数据,Arduino作为子节点,用于发送数据,考虑到以后会有很多子节点,但又不至于使得代码过于繁琐,因此所有的传输数据添加一个头部编号用于区分不同节点. ...

  9. Enterprise Library 3.1 参考源码索引

    http://www.projky.com/entlib/3.1/Microsoft/Practices/EnterpriseLibrary/AppSettings/Configuration/Des ...

  10. vue 实战语法错误导致的问题记录

    1. 在组件中引入mapActions import mapActions  from 'vuex'   而不是  getActions 2.引入 mutations-type.js import   ...