数据结构与算法之从前序与中序遍历序列构造二叉树
问题提出:leetcode原题
给定两个整数数组 preorder 和 inorder ,其中 preorder 是二叉树的先序遍历, inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。
思路分析
视频讲解:这里引用leetcode官方简介视频
[从前序与中序遍历序列构造二叉树 - 从前序与中序遍历序列构造二叉树 - 力扣(LeetCode).ts](assets/从前序与中序遍历序列构造二叉树 - 从前序与中序遍历序列构造二叉树 - 力扣(LeetCode)-20220602150410-cot75wo.ts)
思路总结:
- 第一步:根据前序遍历
A B D E G C F H
确定头结点是A
,根据中序遍历D B G E A C H F
将树划分为左子树D B G E
和右子树C H F
。 - 划分为左右两棵子树:对于左子树,前序遍历是
B D E G
,后续遍历是D B G E
。对于右子树,前序遍历是C F H
,后续遍历是C H F
。 - 对左子树和右子树分别运用第一步和第二步进行分析。
- 递归结束的条件:当中序遍历的节点只剩下一个节点的时候,那么这个节点就是叶子节点。
代码示例
package tree;
import java.util.LinkedHashMap;
/**
* 给定两个整数数组preorder 和 inorder,
* 1. 其中preorder 是二叉树的先序遍历,
* 2. inorder是同一棵树的中序遍历.
* 请构造二叉树并返回其根节点。
*
* @author five-five
* @created 2022/6/2-10:26
*/
public class BuildTreeByMidAndPre {
public TreeNode buildTree(int[] preorder, int[] inorder) {
int preLen = preorder.length;
int inLen = inorder.length;
if (preLen != inLen) {
return null;
}
LinkedHashMap<Integer, Integer> map = new LinkedHashMap<>();
for (int i = 0; i < inorder.length; i++) {
map.put(inorder[i], i);
}
return help(preorder, 0, preLen - 1, map, 0, inLen - 1);
}
/**
* @param preorder 前序遍历数组
* @param preLeft 前序序列左子树开始下标
* @param preRight 前序序列左子树结束下标
* @param map 中序遍历的映射表(提高循环效率)
* @param inLeft 中序序列左子树开始下标
* @param inRight 中序序列左子树结束下标
* @return 返回最后生成的树
*/
private TreeNode help(int[] preorder, int preLeft, int preRight, LinkedHashMap<Integer, Integer> map, int inLeft, int inRight) {
//递归结束条件
if (preLeft > preRight || inLeft > inRight) {
return null;
}
//创建根节点
int rootVal = preorder[preLeft - 1];
TreeNode rootNode = new TreeNode(rootVal);
//根节点在中序遍历序列的位置
int pIndex = map.get(rootVal);
rootNode.left = help(preorder, preLeft + 1, pIndex - inLeft + preLeft, map, inLeft, pIndex - 1);
rootNode.right = help(preorder, pIndex - inLeft + preLeft + 1, preRight, map, pIndex + 1, inRight);
return rootNode;
}
}