七:重建二叉树(依据先序遍历(或者后序遍历)和中序遍历重建二叉树)
2017-06-04 13:11
351 查看
对于一颗二叉树。能够依据先序遍历(或者后序遍历)和中序遍历(树中不含反复的数字)又一次还原出二叉树。
解析:
1. 先序遍历序列的第一个元素必然是根节点,能够由此获取二叉树的根节点。
2. 依据根节点,在中序遍历序列中查找该节点,由中序遍历的性质可知,中序遍历中该根节点左边的序列必然在根节点的左子树中。而根节点右边的序列必然在右子树中。由此能够知道先序遍历中左子树以及右子树的起止位置。
3. 找到了左右子树前序遍历和中序遍历再用相同的方法分别构建左右子树,典型的递归思想。
代码例如以下:
BinaryTreeNode* ConstructCore
(
int* startPreorder, int* endPreorder,
int* startInorder, int* endInorder
)
{
// 前序遍历序列的第一个数字是根结点的值
int rootValue = startPreorder[0];
BinaryTreeNode* root = new BinaryTreeNode();
root->m_nValue = rootValue;
root->m_pLeft = root->m_pRight = NULL;
if(startPreorder == endPreorder)
{
if(startInorder == endInorder && *startPreorder ==*startInorder)
return root;
else
throw std::exception("Invalid input.");
}
// 在中序遍历中找到根结点的值
int* rootInorder = startInorder;
while(rootInorder <= endInorder && *rootInorder != rootValue)
++ rootInorder;
//推断是否找到根节点
if(rootInorder == endInorder && *rootInorder != rootValue)
throw std::exception("Invalid input.");
int leftLength = rootInorder - startInorder;
int* leftPreorderEnd = startPreorder + leftLength;
if(leftLength > 0)
{
// 构建左子树
root->m_pLeft = ConstructCore(startPreorder + 1, leftPreorderEnd,
startInorder, rootInorder - 1);
}
if((leftLength+ startPreorder)< endPreorder ) {
// 构建右子树
root->m_pRight = ConstructCore(leftPreorderEnd + 1, endPreorder,
rootInorder + 1, endInorder);
}
return root;
}
BinaryTreeNode* Construct(int* preorder,int* inorder, int length)
{
if(preorder == NULL || inorder == NULL || length <= 0)
return NULL;
return ConstructCore(preorder, preorder + length - 1,
inorder, inorder + length - 1);
}
注:《剑指offer》学习总结
解析:
1. 先序遍历序列的第一个元素必然是根节点,能够由此获取二叉树的根节点。
2. 依据根节点,在中序遍历序列中查找该节点,由中序遍历的性质可知,中序遍历中该根节点左边的序列必然在根节点的左子树中。而根节点右边的序列必然在右子树中。由此能够知道先序遍历中左子树以及右子树的起止位置。
3. 找到了左右子树前序遍历和中序遍历再用相同的方法分别构建左右子树,典型的递归思想。
代码例如以下:
BinaryTreeNode* ConstructCore
(
int* startPreorder, int* endPreorder,
int* startInorder, int* endInorder
)
{
// 前序遍历序列的第一个数字是根结点的值
int rootValue = startPreorder[0];
BinaryTreeNode* root = new BinaryTreeNode();
root->m_nValue = rootValue;
root->m_pLeft = root->m_pRight = NULL;
if(startPreorder == endPreorder)
{
if(startInorder == endInorder && *startPreorder ==*startInorder)
return root;
else
throw std::exception("Invalid input.");
}
// 在中序遍历中找到根结点的值
int* rootInorder = startInorder;
while(rootInorder <= endInorder && *rootInorder != rootValue)
++ rootInorder;
//推断是否找到根节点
if(rootInorder == endInorder && *rootInorder != rootValue)
throw std::exception("Invalid input.");
int leftLength = rootInorder - startInorder;
int* leftPreorderEnd = startPreorder + leftLength;
if(leftLength > 0)
{
// 构建左子树
root->m_pLeft = ConstructCore(startPreorder + 1, leftPreorderEnd,
startInorder, rootInorder - 1);
}
if((leftLength+ startPreorder)< endPreorder ) {
// 构建右子树
root->m_pRight = ConstructCore(leftPreorderEnd + 1, endPreorder,
rootInorder + 1, endInorder);
}
return root;
}
BinaryTreeNode* Construct(int* preorder,int* inorder, int length)
{
if(preorder == NULL || inorder == NULL || length <= 0)
return NULL;
return ConstructCore(preorder, preorder + length - 1,
inorder, inorder + length - 1);
}
注:《剑指offer》学习总结
相关文章推荐
- 七:重建二叉树(根据先序遍历(或者后序遍历)和中序遍历重建二叉树)
- 刷题之路----根据前序遍历和中序遍历或者后序遍历和中序遍历重建二叉树
- java-从先序遍历和中序遍历重建二叉树
- 已知二叉树的先序遍历序列和中序遍历序列,输出该二叉树的后序遍历序列
- poj2255 根据任意二叉树的先序遍历和中序遍历求解后序遍历
- 已知先序遍历和中序遍历,求后序遍历 && 求二叉树中节点的最大距离
- c/c++实现利用二叉树的先序遍历和中序遍历序列重建树
- 输入前序/后序+中序 遍历结果重建二叉树(递归)
- 已知二叉树的先序遍历序列和中序遍历序列,输出该二叉树的后序遍历序列
- java-从先序遍历和中序遍历重建二叉树
- [leetcode]_根据二叉树的先序遍历(后序遍历) + 中序遍历 重建二叉树
- 二叉树有先序遍历和中序遍历,构造出后序遍历
- 根据二叉树前序遍历和中序遍历的结果,重建出该二叉树并后序遍历之 java代码
- 先序遍历序列和中序遍历序列,输出该二叉树的后序遍历序列
- 1935. 二叉树重建(根据先序遍历和中序遍历重建二叉树并进行广度优先遍历)
- 根据后序和中序遍历重建二叉树
- 根据前序遍历(或者是后序遍历)和中序遍历得到的序列可以重建二叉树
- 已知二叉树的先序遍历序列和中序遍历序列,输出该二叉树的后序遍历序列
- 二叉树根据先序遍历和中序遍历,得到后序遍历
- 根据先序遍历数列和中序遍历数列重建二叉树