6.3.3 二叉树重建 已知前序和中序求后序

在一棵二叉树总,前序遍历结果为:DBACEGF,中序遍历结果为 ABCDEFG,求后序遍历结果。

我们知道:

前序遍历方式为:根节点->左子树->右子树

中序遍历方式为:左子树->根节点->右子树

后序遍历方式为:左子树->右子树->根节点

从这里可以看出,前序遍历的第一个值就是根节点,然后再中序遍历中找到这个值,那么这个值的左边部分即为当前二叉树的左子树部分前序遍历结果,这个值的右边部分即为当前二叉树的右子树部分前序遍历结果。因此,通过这个分析,可以恢复这棵二叉树,得到这样的一段伪码:


节点 getRoot(前序,中序)

c=前序第一个字符

pos=c在中序中的位置

len1=中序pos左半部分长度

len2=中序pos右半部分长度

新建节点r,令r的元素等于c

r的左儿子=getRoot(前序位置1开始的len1长度部分,中序pos位置的左半部分)

r的右儿子=getRoot(前序位置len1开始右半部分,中序pos位置的右半部分)

return r

下面有两种求法:

1.不建立树,直接输出结果

#include<stdio.h>
#include<string.h>

char ans[8];
char *S1 = "DBACEGF";
char *S2 = "ABCDEFG";


void build(int n, char *s1, char *s2, char *s)
{
	if(n <= 0)
		return;
	int p = strchr(s2,s1[0]) - s2;
	build(p,s1+1,s2,s);
	build(n-p-1,s1+p+1,s2+p+1,s+p);
	s[n-1] = s1[0];
} 

int main()
{
	int n = strlen(S1);
	build(n,S1,S2,ans);
	ans[n] = '\0';
	printf("%s\n",ans);
}

2.建立二叉树

#include <iostream>
#include <fstream>
#include <string>
using namespace std;

struct TreeNode
{
  struct TreeNode* left;
  struct TreeNode* right;
  char  elem;
};

TreeNode* BinaryTreeFromOrderings(char* inorder, char* preorder, int length)
{
  if(length == 0)
  		return NULL;
  TreeNode* node = new TreeNode;//Noice that [new] should be written out.
  node->elem = *preorder;
  node->left = node ->right = NULL;
  int rootIndex = 0;
  for(;rootIndex < length; rootIndex++)//a variation of the loop
  {
      if(inorder[rootIndex] == *preorder)
      	break;
  }
  node->left = BinaryTreeFromOrderings(inorder, preorder +1, rootIndex);
  node->right = BinaryTreeFromOrderings(inorder + rootIndex + 1, preorder + rootIndex + 1, length - (rootIndex + 1));
  cout << node->elem << " ";
  return node;
}
int main()
{
    char* pr="DBACEGF";    
 	char* in="ABCDEFG"; 
	BinaryTreeFromOrderings(in, pr, 7); 
	printf("\n"); 
	return 0;
}


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值