题目:

给定一个没有重复元素的数组A,定义A上的MaxTree如下:MaxTree的根节点为A中最大的数,根节点的左子树为数组中最大数左边部分的MaxTree,右子树为数组中最大数右边部分的MaxTree。请根据给定的数组A,设计一个算法构造这个数组的MaxTree。

思路:

如果能够确定每个节点的父亲节点,则可以构造出整棵树。找出每个数往左数第一个比他大的数和往右数第一个比他大的数,两者中较小的数即为该数的父亲节点。如:[3,1,2],3没有父亲节点,1的父亲节点为2,2的父亲节为3。并且可以根据与父亲的位置关系来确定是左儿子还是右儿子。接下来的问题是如何快速找出每个数往左、往右第一个比他大的数。这里需要用到数据结构栈。以找每个数左边第一个比他大的数为例,从左到右遍历每个数,栈中保持递减序列,新来的数不停的Pop出栈顶直到栈顶比新数大或没有数。以[3,1,2]为例,首先3入栈,接下来1比3小,无需pop出3,1入栈,并且确定了1往左第一个比他大的数为3。接下来2比1大,1出栈,2比3小,2入栈。并且确定了2往左第一个比他大的数为3。用同样的方法可以求得每个数往右第一个比他大的数。时间复杂度O(n),空间复杂度也是O(n)为最优解法。

代码:

#include <iostream>
#include <vector>
#include <stack> using namespace std; struct Node{
int val;
int idx;
Node *left;
Node *right;
Node(int v,int i):val(v),idx(i),left(NULL),right(NULL){}
}; void PreOrderTraverse(Node* root){
if(root!=NULL){
cout<< root->val <<" ";
PreOrderTraverse(root->left);
PreOrderTraverse(root->right);
}
} Node* MaxTree(const vector<int> &A,int n){
stack<Node*> leftStk;
stack<Node*> rightStk;
vector<Node*> tree(n);
vector<int> lMax(n);
vector<int> rMax(n); for(int i=;i<n;i++)
tree[i]=new Node(A[i],i); for(int i=;i<n;i++){
if(!leftStk.empty()){
while(!leftStk.empty() && leftStk.top()->val<A[i])
leftStk.pop();
if(!leftStk.empty())
lMax[i]=leftStk.top()->idx;
else
lMax[i]=-;
}
else
lMax[i]=-;
leftStk.push(tree[i]);
} for(int i=n-;i>=;i--){
if(!rightStk.empty()){
while(!rightStk.empty() && rightStk.top()->val<A[i])
rightStk.pop();
if(!rightStk.empty())
rMax[i]=rightStk.top()->idx;
else
rMax[i]=-;
}
else
rMax[i]=-;
rightStk.push(tree[i]);
} int root=;
for(int i=;i<n;i++){
if(lMax[i]==- && rMax[i]==-){
root=i;
continue;
} int parent;
if(lMax[i]==-)
parent=rMax[i];
else if(rMax[i]==-)
parent=lMax[i];
else
parent=A[lMax[i]]<A[rMax[i]]?lMax[i]:rMax[i]; if(i<parent)
tree[parent]->left=tree[i];
else
tree[parent]->right=tree[i];
} /*
for(int i=0;i<n;i++){
cout<<tree[i]->idx <<":";
if(tree[i]->left)
cout<<"left: "<<tree[i]->left->idx<<" ";
if(tree[i]->right)
cout<<"right:"<<tree[i]->right->idx;
cout<<endl;
}
*/
return tree[root];
} int main()
{
int n;
while(cin>>n){
vector<int> A(n);
for(int i=;i<n;i++)
cin>>A[i];
Node* root=MaxTree(A,n);
PreOrderTraverse(root);
cout<<endl;
}
return ;
}

(算法)构造MaxTree的更多相关文章

  1. 构造MaxTree

    链接:https://www.nowcoder.com/questionTerminal/a502c7c3c65e41fdaf65eec9e0654dcb 来源:牛客网 [编程题]构造MaxTree ...

  2. Kruskal算法构造最小生成树

    Kruskal算法来构造最小生成树,我总结了分为以下步骤: (1)建图,构造Kruskal边集,边集元素应该包括该边的起始顶点.终止顶点.权值: (2)将边集按权值从小到大的顺序进行排序: (3)从小 ...

  3. c/c++ 用克鲁斯卡尔(kruskal)算法构造最小生成树

    c/c++ 用克鲁斯卡尔(kruskal)算法构造最小生成树 最小生成树(Minimum Cost Spanning Tree)的概念: 假设要在n个城市之间建立公路,则连通n个城市只需要n-1条线路 ...

  4. c/c++ 用普利姆(prim)算法构造最小生成树

    c/c++ 用普利姆(prim)算法构造最小生成树 最小生成树(Minimum Cost Spanning Tree)的概念: ​ 假设要在n个城市之间建立公路,则连通n个城市只需要n-1条线路.这时 ...

  5. 图的建立(邻接矩阵)+深度优先遍历+广度优先遍历+Prim算法构造最小生成树(Java语言描述)

    主要参考资料:数据结构(C语言版)严蔚敏   ,http://blog.chinaunix.net/uid-25324849-id-2182922.html   代码测试通过. package 图的建 ...

  6. 【Machine Learning in Action --2】K-近邻算法构造手写识别系统

    为了简单起见,这里构造的系统只能识别数字0到9,需要识别的数字已经使用图形处理软件,处理成具有相同的色彩和大小:宽高是32像素的黑白图像.尽管采用文本格式存储图像不能有效地利用内存空间,但是为了方便理 ...

  7. Dijkstra算法构造单源点最短路径

    迪杰斯特拉(Dijkstra)算法 是求从某个源点到其余各顶点的最短路径,即对已知图 G=(V,E),给定源顶点 s∈V,找出 s 到图中其它各顶点的最短路径. 我总结下核心算法,伪代码如下: Dij ...

  8. 《程序员代码面试指南》第一章 栈和队列 构造数组的MaxTree

    题目 给出一个无重复元素的数组,构造此数组的MaxTree, java代码 /** * @Description: 构造数组的MaxTree * @Author: lizhouwei * @Creat ...

  9. 数据挖掘算法(一)C4.5

    统计了14天的气象数据D(指标包括outlook,temperature,humidity,windy),并已知这些天气是否打球(play).如果给出新一天的气象指标数据:sunny,cool,hig ...

随机推荐

  1. [BZOJ5306][HAOI2018]染色(容斥+FFT)

    https://www.cnblogs.com/zhoushuyu/p/9138251.html 注意如果一开始F(i)中内层式子中j枚举的是除前i种颜色之外还有几种出现S次的颜色,那么后面式子就会难 ...

  2. 极域电子教室3个没被发现的bug(V6 2007)

    bug1:破坏文件传输 在教师端文件分发时利用各种方法退出极域电子教室(包括http://sunnysab.blog.163.com/blog/static/180375009201323191111 ...

  3. hdu 5289 Assignment(2015多校第一场第2题)RMQ+二分(或者multiset模拟过程)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5289 题意:给你n个数和k,求有多少的区间使得区间内部任意两个数的差值小于k,输出符合要求的区间个数 ...

  4. hdu 2121

    朱刘算法求无根最小树形图 可以任意选一个根,求最小的权和以及当时的根. 先建一个超级根,它连向所有点,边权为所有边的边权和加1(即sumw+1),然后求以它为根的最小树形图,再根据树形图权和与2*(s ...

  5. 趁热打铁学node

    前言 不废话直接官网下载安装. windows安装很简单,双击ok. 完成后命令行输入node -v. 会提示版本号,说明安装成功(Node.js 还自动附带安装了 npm,类似ruby的gem). ...

  6. 模仿JQuery 的添加多个事件的原理

    var jquery=function(dom){ var obj={ ready:function(fn){ if(typeof dom.onload=="function"){ ...

  7. Tasker to answer incoming call by pressing power button

    nowadays, the smartphone is getting bigger in size, eg. samsung galaxy note and note 2, sorta big in ...

  8. GDB 调试PYTHON

    http://www.cnblogs.com/dkblog/p/3806277.html

  9. as 汇编器

    [root@localhost ~]# cat .s .file "write.s" .section .rodata hello: .string "hello, wo ...

  10. ldd pvs dis on solaris 10

    #include <QtGui/QApplication> #include <QtGui/QDialog> int main(int argc, char *argv[]) ...