[剑指offer] 重建二叉树,根据前中,输出后,根据中后,输出前

参考:《剑指offer》纪念版

情况1.:给出树的前序序列和中序序列,输出后序序列

情况2 :给出树的后序序列和中序序列,输出前序序列

解决方法:根据所给出的两个序列,构造出(重建)二叉树,然后按要求再遍历输出。

重建二叉树主要利用了递归的思想,最重要的是找出序列的范围(函数传参),这个要非常仔细,很容易出错。一定要在纸上画出图,然后根据图来确定范围

用到的两个函数:

以情况1为例:

主要用到了两个函数:

BinaryTreeNode* Construct(char* preorder, char* inorder, int length)//给出前序序列,中序序列,长度,重建二叉树
BinaryTreeNode* ConstructCore
(
     char* startPreorder, char* endPreorder,
     char* startInorder, char* endInorder
)//前序序列起始位置,中序序列起始位置

给出前序序列和中序序列,直接调用第一个函数就可以。

完整的代码:

#include <iostream>
#include <stdio.h>
#include <algorithm>
#include <string.h>
#include <set>
#include <map>
#include <stack>
#include <queue>
using namespace std;

struct BinaryTreeNode
{
    char               m_nValue;
    BinaryTreeNode*    m_pLeft;
    BinaryTreeNode*    m_pRight;
};

bool invalidInput;//输入的序列是否合法

void Visit(BinaryTreeNode* node)//访问节点
{
    cout << node->m_nValue;
}

void PreorderTraverse(BinaryTreeNode* root)//前序遍历
{
    if(root == NULL)
        return;
    Visit(root);
    PreorderTraverse(root->m_pLeft);
    PreorderTraverse(root->m_pRight);
}

void InorderTraverse(BinaryTreeNode* root)//中序遍历
{
    if(root == NULL)
        return;
    InorderTraverse(root->m_pLeft);
    Visit(root);
    InorderTraverse(root->m_pRight);
}

void PostorderTraverse(BinaryTreeNode* root)//后序遍历
{
    if(root == NULL)
        return;
    PostorderTraverse(root->m_pLeft);
    PostorderTraverse(root->m_pRight);
    Visit(root);
}

BinaryTreeNode* ConstructCore
(
     char* startPreorder, char* endPreorder,
     char* startInorder, char* endInorder
)//前序序列起始位置,中序序列起始位置
{
    BinaryTreeNode* root = (BinaryTreeNode* )malloc(sizeof(BinaryTreeNode));
    char rootValue = startPreorder[0];//根节点的值为前序遍历序列的第一个值
    root->m_nValue = rootValue;
    root->m_pLeft = root->m_pRight = NULL;

    if(startPreorder == endPreorder)
    {
        if(startInorder == endInorder && *startPreorder == *startInorder)//如果前序序列和中序序列都只有一个字母,且二者字母相同
            return root;
        else
        {
            invalidInput = true;
            return NULL;
        }
    }

    //在中序序列中找到根节点的位置
    char* rootInorder = startInorder;//根节点在在中序序列中的位置
    while(rootInorder <= endInorder && (*rootInorder) != rootValue)
        ++ rootInorder;

    if(rootInorder > endInorder)//找不到根节点
    {
        invalidInput = true;
        return NULL;
    }
    char leftLength = rootInorder - startInorder;//左子树的长度
    char* leftPreorderEnd = startPreorder + leftLength;//左子树在前序序列中的终点位置
    if(leftLength > 0)//构造左子树
    {
        root->m_pLeft = ConstructCore(startPreorder + 1, leftPreorderEnd, startInorder, rootInorder - 1);
    }

    char rightLength = endPreorder - startPreorder - leftLength;
    if(rightLength > 0)//构造右子树
    {
        root->m_pRight = ConstructCore(leftPreorderEnd + 1, endPreorder, rootInorder + 1, endInorder);
    }
    return root;
}

BinaryTreeNode* Construct(char* preorder, char* inorder, int length)//给出前序序列,中序序列,长度,重建二叉树
{
    if(preorder == NULL || inorder == NULL || length < 0)
        return NULL;
    return ConstructCore(preorder, preorder + length -1, inorder, inorder + length -1);
}

char preOrder[50], inOrder[50];
int main()
{
    while(cin>> preOrder >> inOrder)
    {
        invalidInput = false;
        int len = strlen(preOrder);
        BinaryTreeNode* root = Construct(preOrder, inOrder, len);
        if(invalidInput == true)
        {
            cout <<"Invalid input." <<endl;
            continue;
        }

        cout << "前序遍历为: " ;
        PreorderTraverse(root);
        cout << endl;

        cout << "中序遍历为: " ;
        InorderTraverse(root);
        cout << endl;

        cout << "后序遍历为: " ;
        PostorderTraverse(root);
        cout << endl;
    }

    return 0;
}

情况2和情况1很类似,也是利用递归,重要的也是找准序列的范围,下面给出关键代码:

BinaryTreeNode* ConstructCore2
(
 char* startPostorder, char* endPostorder,
 char* startInorder, char* endInorder
)//后序序列起始位置,终点位置, 中序序列起始位置,终点位置
{
    BinaryTreeNode* root = (BinaryTreeNode*) malloc(sizeof(BinaryTreeNode));
    root->m_nValue = (*endPostorder);//根节点为后序序列的最后一个值
    root->m_pLeft = root->m_pRight = NULL;
    int rootValue = root->m_nValue;

    if(startPostorder == endPostorder)
    {
        if(startInorder == endInorder && *startPostorder == *startInorder)//如果后序序列和中序序列都只有一个节点,且节点值相等
            return root;
        else
        {
            invalidInput = true;
            return NULL;
        }
    }

    //在中序序列中找到根节点的位置
    char* rootInorder = startInorder;
    while(rootInorder <= endInorder && *rootInorder != rootValue)
        ++ rootInorder;
    if(rootInorder > endInorder)//没找到
    {
        invalidInput = true;
        return NULL;
    }

    //构造左子树
    int leftLength = rootInorder - startInorder;
    if(leftLength > 0)
    {
        root->m_pLeft = ConstructCore2(startPostorder, startPostorder + leftLength - 1, startInorder, rootInorder - 1);//这种参数看着图写
    }

    int rightLength = endInorder - startInorder - leftLength;
    if(rightLength > 0)
    {
        root->m_pRight = ConstructCore2(endPostorder - rightLength, endPostorder - 1, rootInorder + 1, endInorder);
    }
    return root;

}

BinaryTreeNode* Construct2(char* postorder,char* inorder, int length)//给出后序序列,中序序列,长度,重建二叉树
{
    if(postorder == NULL || inorder == NULL || length < 0)
        return NULL;
    return ConstructCore2(postorder, postorder + length -1, inorder, inorder + length - 1);
}

P.S: 书中57页 在中序遍历中找到根节点值 下面的这几行代码:

int* rootInorder = startInorder;
while(rootInorder <= endInorder && *rootInorder != rootValue)
    ++rootInorder;

if(rootInorder == endInorder && *rootInorder != rootValue)
    throw std::exception("Invalid input.");

感觉有问题。 下面为什么还要判断 rootInorder == endInorder ?

时间: 2024-12-21 13:05:05

[剑指offer] 重建二叉树,根据前中,输出后,根据中后,输出前的相关文章

剑指offer重建二叉树python

题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回. 思路 前序遍历第一个节点是根节点,该节点在中序遍历中的前部分是左子,后部分是右子,将左子部分右子部分重新重复这个判断过程即可. 代码 # -*- coding:utf-8 -*- # class TreeNode: # def __init__(sel

剑指offer 重建二叉树

重建二叉树 题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回 思路:首先找到root,然后递归的重建root -> left,root -> right. /** * Definition for binary tree * struct TreeNode { * int val; * TreeNod

剑指Offer -- 重建二叉树(详细思路)

题目描述: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字. 例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回. 先贴上完整代码: 1 /** 2 * Definition for binary tree 3 * public class TreeNode { 4 * int val; 5 * TreeNode left; 6 * TreeNode right

剑指offer——重建二叉树

题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回. # -*- coding:utf-8 -*- # class TreeNode: # def __init__(self, x): # self.val = x # self.left = None # self.right = None class

剑指offer ——重建二叉树

p62 输入前序和中序遍历的结果(不包含重复的数字),重建二叉树. 主要是分析两个序列的规律,然后用递归的方式找到每一个根节点,从而完成构建. #include<iostream> #include <vector> #include <set> #include <functional> #include <stdlib.h> #include <stdio.h> #include <string> #include &

【剑指offer】二叉树的镜像

转载请注明出处:http://blog.csdn.net/ns_code/article/details/25915971 题目描述: 输入一个二叉树,输出其镜像. 输入: 输入可能包含多个测试样例,输入以EOF结束.对于每个测试案例,输入的第一行为一个整数n(0<=n<=1000,n代表将要输入的二叉树节点的个数(节点从1开始编号).接下来一行有n个数字,代表第i个二叉树节点的元素的值.接下来有n行,每行有一个字母Ci.Ci='d'表示第i个节点有两子孩子,紧接着是左孩子编号和右孩子编号.C

剑指OFFER之二叉树的镜像(九度OJ1521)

题目描述: 输入一个二叉树,输出其镜像. 输入: 输入可能包含多个测试样例,输入以EOF结束.对于每个测试案例,输入的第一行为一个整数n(0<=n<=1000,n代表将要输入的二叉树节点的个数(节点从1开始编号).接下来一行有n个数字,代表第i个二叉树节点的元素的值.接下来有n行,每行有一个字母Ci.Ci='d'表示第i个节点有两子孩子,紧接着是左孩子编号和右孩子编号.Ci='l'表示第i个节点有一个左孩子,紧接着是左孩子的编号.Ci='r'表示第i个节点有一个右孩子,紧接着是右孩子的编号.C

剑指offer (19) 二叉树镜像 对称二叉树

题目: 输入一个二叉树,输出其镜像. BinTreeNode* ReverseTree(BinTreeNode* pRoot) { if (pRoot == NULL) return NULL; BinTreeNode* pLeftReverse = ReverseTree(pRoot->left); BinTreeNode* pRightReverse = ReverseTree(pRoot->right); pRoot->left = pRightReverse; pRoot->

剑指offer (39) 二叉树深度

题目:输入一棵二叉树的根节点,求该树的深度 题解分析: 二叉树具有天然的递归性,首先应该想递归解法 /** * Definition for binary tree * struct TreeNode { * int val; * TreeNode *left; * TreeNode *right; * TreeNode(int x) : val(x), left(NULL), right(NULL) {} * }; */ class Solution { public: int maxDept