博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Construct Binary Tree from Preorder and Inorder Traversal -- LeetCode
阅读量:5070 次
发布时间:2019-06-12

本文共 1235 字,大约阅读时间需要 4 分钟。

原题链接: 
 

这道题是树中比較有难度的题目。须要依据先序遍历和中序遍历来构造出树来。这道题看似毫无头绪,事实上梳理一下还是有章可循的。以下我们就用一个样例来解释怎样构造出树。

如果树的先序遍历是12453687。中序遍历是42516837。

这里最重要的一点就是先序遍历能够提供根的所在,而依据中序遍历的性质知道根的所在就能够将序列分为左右子树。比方上述样例,我们知道1是根。所以依据中序遍历的结果425是左子树,而6837就是右子树。

接下来依据切出来的左右子树的长度又能够在先序便利中确定左右子树相应的子序列(先序遍历也是先左子树后右子树)。

依据这个流程,左子树的先序遍历和中序遍历各自是245和425。右子树的先序遍历和中序遍历则是3687和6837,我们反复以上方法,能够继续找到根和左右子树。直到剩下一个元素。能够看出这是一个比較明显的递归过程。对于寻找根所相应的下标。我们能够先建立一个HashMap,以免后面须要进行线行搜索。这样每次递归中就仅仅须要常量操作就能够完毕对根的确定和左右子树的切割。


算法终于相当于一次树的遍历,每一个结点仅仅会被訪问一次,所以时间复杂度是O(n)。而空间我们须要建立一个map来存储元素到下标的映射,所以是O(n)。

代码例如以下:

public TreeNode buildTree(int[] preorder, int[] inorder) {    if(preorder==null || inorder==null)        return null;    HashMap
map = new HashMap
(); for(int i=0;i
map){ if(preL>preR || inL>inR) return null; TreeNode root = new TreeNode(preorder[preL]); int index = map.get(root.val); root.left = helper(preorder, preL+1, index-inL+preL, inorder, inL, index-1, map); root.right = helper(preorder, preL+index-inL+1, preR, inorder, index+1, inR,map); return root;}
能够看出上面实现结果还是很接近于一次树的遍历的,仅仅是我们是以一个构造树的形式,在遍历中把树创建出来。这样的题目算是树中的难题了,只是理清思路,事实上也只是如此哈~

转载于:https://www.cnblogs.com/ldxsuanfa/p/9941872.html

你可能感兴趣的文章
对数据库中的表或字段重命名
查看>>
Alpha阶段项目总结
查看>>
正则表达式(进阶篇)
查看>>
你知道的和不知道的sass
查看>>
#ifndef/#define/#endif以及#if defined/#else/#endif使用详解
查看>>
实例分析如何精确C#日期格式到毫秒,DateTime调用
查看>>
小记 百度地图 soso地图 经纬度偏移
查看>>
【C】由printf("%d\t%d\t%d\n",a,a+=(a++),a);引起的思考
查看>>
Eclipse与Android源码中ProGuard工具的使用
查看>>
阿里云服务器ecs配置之安装mysql
查看>>
POJ 2484 A Funny Game【博弈】
查看>>
字符串及其操作,字符的Unicode编码
查看>>
制作图片识别文字软件
查看>>
IP地址中的保留地址
查看>>
[转载]细说ASP.NET Windows身份认证
查看>>
salt-api配置安装 以及使用
查看>>
NHibernate.3.0.Cookbook第四章第7节的翻译
查看>>
Docker 监控平台Prometheus
查看>>
vertical-align 和line-height 以及baseline的解析
查看>>
Android学习第十四天----SAX解析xml
查看>>