codility上的问题 (21) Upsilon 2012
这是我目前最喜欢的codiltiy上的问题之一。问题描述是:给定一个整数数组A,所有的数均不相同。假设下标从0开始,找到一个数组B, 满足A[B[0]] > A[B[1]] > A[B[2]] >...A[B[K]],对任意两项A[B[i]]和A[B[i + 1]],任意j, min(B[i],B[i + 1]) < j < max(B[i],B[i + 1]) 均有A[j] < A[B[i + 1]] ,求最大的K。
例如,对数组 A,
A[0] = 9 A[1] = 10 A[2] = 2
A[3] = -1 A[4] = 3 A[5] = -5
A[6] = 0 A[7] = -3 A[8] = 1
A[9] = 12 A[10] = 5 A[11] = 8
A[12] = -2 A[13] = 6 A[14] = 4
求得的数组B为 9,1,4,8,6,7,长度为6
A[9] = 12 A[1] = 10 A[4] = 3
A[8] = 1 A[6] = 0 A[7] = -3
输入数组A的长度n [1..10^5],数组元素范围[-10^9,+10^9],都是整数且不相同。
要求复杂度 时间空间都是O(n)。
分析: 这个题乍一看没思路。需要转换:我们把A中得数建成一个类似”堆“的结构。对A数组,这个二叉树的根是最大值,然后我们把最大值两边的部分分别建立成这样的二叉树(每段递归的找最大值)。我们暂时抛开建树的复杂度,如果有了这样的树,我们所以的B数组长度对应为从树根到叶子的叶子的一条路径长度。(因为每个子树的根都是最大值)。那么如何建立这样的树呢? 暴力递归的方法会导致O(n^2)的复杂度。如果我们从做到右扫描A数组,假设某个值是当前最大值,它的左子树所有的元素都确定了,再出现的数,要么放在它的右子树,要么它成为别人的左子树。对于任意一个元素来讲,要么它成为之前元素的右子树,要么之前某个元素成为它的左子树,这取决于大小关系。我们在算一个元素的右子树的时候,因为元素还没扫描完,所以我们很难确定什么时候把它作为某个元素的右子树位置。 我们可以保存这样一个栈,栈的每个元素是一棵树,树根的左子树已经完全确定,根的右子树暂时是空。当弹出一棵树的时候,我们需要把它下面那个元素的子树合并。
也就是说,栈考顶得元素是其下面元素的右子树,但是右子树会变化。
大概过程如下:
如果当前元素比栈顶元素大,就把栈顶元素弹出,新弹出的元素作为之前弹出元素的右子树,都弹出之后形成的树,作为当前元素的左子树(右子树为空),形成的树放入栈。因为弹出来的东西都比当前元素小,并且在当前元素的左边。另外,我们不会真得合并树,只记录树的高度即可(根到叶子的节点个数)。代码非常简单:
// you can also use includes, for example:
// #include <algorithm>
int solution(const vector<int> &A) {
// write your code here...
int i, height, n = A.size();
vector<pair<int,int> > s;
for (i = 0; i < n; ++i) {
for (height = 0; (!s.empty()) && (s.back().first < A[i]);s.pop_back()) {
height = max(height + 1, s.back().second);
}
s.push_back(make_pair(A[i], height + 1));
}
for (height = 0; !s.empty();s.pop_back()) {
height = max(height + 1, s.back().second);
}
return height;
}
codility上的问题 (21) Upsilon 2012的更多相关文章
- codility上的问题(15) Xi 2012
进入2012年的题 codility上的题目开始变难,变得有意思起来.给定两个长度在[1..300000]的只包含0和1的串S和T,它们是2进制表示的,S表示的数A不大于T表示的数B,即A<=B ...
- codility上的练习 (1)
codility上面添加了教程.目前只有lesson 1,讲复杂度的……里面有几个题, 目前感觉题库的题简单. tasks: Frog-Jmp: 一只青蛙,要从X跳到Y或者大于等于Y的地方,每次跳的距 ...
- codility上的问题 (19)Sigma 2012
题目: 像最大直方图一样给定一个数组是每个单位长度上的高度,求至少几个矩形可以拼出这个形状. 例如:给出的数组 H[0] = 8 H[1] = 8 H[2] = 5 H[3] = 7 H[4] = 9 ...
- codility上的问题(18) Rho 2012
从正整数1开始,产生一个数列,数列中的每个数是之前出现过的任意两个数的和(可以相等),问产生正整数A,需要的数列长度至少是多少?返回这样一个最短的序列. 例如A=42 可以这样[1, 2, 3, 6, ...
- codility上的问题 (23)Chi 2012
这个题也比较有意思.意思是给定一个数组A,长度为M,里面都是正整数,代表每块地形的高度.现在要测试一种加农炮,给定一个炮弹的高度H, 如果存在最小的I,满足0 < I < M,满足A[I ...
- Codility上的问题 (16) Omicron 2012
比较无聊的题,求斐波那契数的第N^M项. f(0) = 0, f(1) = 1, f(n) = f(n - 1) + f(n - 2),结果对10000103取模. N, M在[0..10^7]之间. ...
- Codility上的问题 (17) PI 2012
这个题比较简单,给定一个整数数组,对每个元素,求出和它最近比它大的数的距离(下标绝对值),如果没有比它大的数,认为距离是0. 数组元素个数 N [0..50000],数组元素范围[-10^9, +10 ...
- codility上的练习(5)
codility出了lesson 5了. (1) 合法括号序列,包括( [ { ) ] }这6种字符的字符串,长度N在[0..200000]范围内,为其是否合法. 要求时间复杂度O(N),空间复杂度O ...
- codility上的问题(34) Fluorum 2014
好久没写codility的题了.一来没时间,二来有的题目不太好分析.这个题比較有意思,我还没有给出很严格的证明.
随机推荐
- BZOJ2023: [Usaco2005 Nov]Ant Counting 数蚂蚁
2023: [Usaco2005 Nov]Ant Counting 数蚂蚁 Time Limit: 4 Sec Memory Limit: 64 MBSubmit: 56 Solved: 16[S ...
- CH Round #53-数据备份
描述 已知有N座办公楼位于同一条街上.你决定给这些办公楼配对(两个一组).每一对办公楼可以通过在这两个建筑物之间铺设网络电缆使得它们可以互相备 份.然而,网络电缆的费用很高.当地电信公司仅能为你提供K ...
- jQuery 1.6+ 中attr()与prop() 区别
最近在写一个关于checkbox全选与取消全选的优化方法时,看到很多高手用到了.prop(). 于是在jquery的帮助文档查了一下,才知道这是在jquery 1.6.1中新加的方法,用来设置属性.但 ...
- 第34讲 UI组件之 ProgressDialog和Message
第34讲UI组件之 ProgressDialog和Message 1.进度对话框 ProgressDialog <1>简介 ProgressDialog是AlertDialog类的一个扩展 ...
- java 读取mysql库表数据
import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import ...
- Spark Accumulators
概述 Accumulator即累加器,与Mapreduce counter的应用场景差不多,都能很好地观察task在运行期间的数据变化,Spark中的Accumulator各task可以对Accumu ...
- JavaScripts学习日记——BOM
IE 3.0 和 Netscape Navigator 3.0 提供了一种特性 - BOM(浏览器对象模型),可以对浏览器窗口进行访问和操作.使用 BOM,开发者可以移动窗口.改变状态栏中的文本以及执 ...
- 解决linux不能使用chmod更改权限的问题
本人安装的是win10和ubuntu的双系统,发现在ubuntu下挂载windows硬盘不用命令chmod更改文件的权限,解决方法记录如下: 对于使用命令$ chmod 777 dirname更改不了 ...
- DTO学习系列之AutoMapper(五)----当EntityFramework爱上AutoMapper
有时候相识即是一种缘分,相爱也不需要太多的理由,一个眼神足矣,当EntityFramework遇上AutoMapper,就是如此,恋爱虽易,相处不易. 在DDD(领域驱动设计)中,使用AutoMapp ...
- oracle RAC调整数据文件大小并移动表到指定的表空间
一.Oracle RAC 调整表空间数据文件大小 1.先查找出表空间对应的数据文件路径: select file_name,tablespace_name from dba_data_files ; ...