【剑指Offer_4】输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序

题目:输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回。

解题思路:

前序遍历的顺序是 根---左结点---右结点。

中序遍历的顺序是 左结点---根---右结点。

因此,对于前序遍历序列,其实就是“根”的遍历,1是第一个遍历到的根,2是第二个会遍历的根,4是第三个……

而对于中序遍历序列,对于根节点1来说,它的左边全部为左子树【4,7,2】,右边全部为它的右子树【5,3,8,6】,而在左子树中,如何判断谁是根节点呢,看前序遍历序列的下一个结点“2”,因为对于1来说,它的左子树的根节点,肯定是前序遍历中1的下一个结点。因此,对于以结点2为根节点的子树,左边的【4,7】为它的左子树,而右边没有子树……以此类推出整个树的结构。

代码可以这样写:

1)用两个下标在中序遍历序列中控制遍历的数组范围(因为每次只遍历某一颗子树),比如下标x=0,下标y=3,则只遍历根节点为1的左子树;

2)在前序遍历序列中,用一个pre_index来控制根节点的遍历位置,每遍历过一个子树根节点,pre_index++。

完整代码如下:

public class Solution {
   private int [] pre;  //前序遍历数组
	private int pre_index = 0;   //用来遍历前序遍历数组的下标
	
	public TreeNode LeftNodeAndRightNode( int x, int y, int shuzu[] ) {   //主递归方法,x左边界,y右边界,shuzu中序遍历数组
		int index = Find ( pre[ this.pre_index++ ], x, y, shuzu ); //调用Find方法判断下一个根节点pre[]在中序数组shuzu中的下标
		TreeNode temp = new TreeNode( shuzu[index] ); //新建子树根结点
		if ( index != x ) {  //如果存在左子树
			temp.left = LeftNodeAndRightNode( x, index-1, shuzu );  //递归,寻找左子树
		}
		if( index != y ) {  //如果存在右子树
			temp.right = LeftNodeAndRightNode( index+1, y, shuzu );  //递归,寻找右子树
		}
		return temp;
	}
	
	public int Find( int target, int x, int y, int shuzu[] ) {  //返回target在数组shuzu中的位置
		int i;
        for ( i = x; i <= y; i++ ) {
        	if ( shuzu[i] == target ) {
        		break;
        	}
        }
        return i;
	}
	
	public TreeNode reConstructBinaryTree(int [] pre,int [] in) {  //被main调用的方法,pre前序遍历,in中序遍历
		TreeNode tn = new TreeNode( pre[0] );  //创建整棵树的根节点
		this.pre = pre;
		int in_len = in.length;
        int rootNode_index = Find( pre[ pre_index++ ],0, in_len, in );  //找到整棵树的根节点在中序遍历序列中的位置
        if ( rootNode_index != 0 ) {  //如果存在左子树
        	tn.left = LeftNodeAndRightNode( 0, rootNode_index-1, in );  //递归,寻找左子树
        }
        if ( rootNode_index != in_len-1 ) {   //如果存在右子树
        	tn.right = LeftNodeAndRightNode( rootNode_index+1, in_len-1, in );  //递归,寻找右子树
        }
        return tn;
    }
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以通过前序遍历中序遍历的结果重建二叉树,然后对重建后的二叉树进行后遍历,即可得到所求的后遍历结果。具体步骤如下: 1. 根据前序遍历的结果确定二叉树的根节点。 2. 在中序遍历的结果找到根节点的位置,将中序遍历结果分为左子树和右子树两部分。 3. 根据左子树的前序遍历中序遍历结果递归重建左子树。 4. 根据右子树的前序遍历中序遍历结果递归重建右子树。 5. 将左子树和右子树连接到根节点上,得到重建后的二叉树。 6. 对重建后的二叉树进行后遍历,得到所求的后遍历结果。 需要注意的是,如果前序遍历中序遍历结果存在重复元素,那么需要根据前序遍历顺序来确定左子树和右子树的范围。 ### 回答2: 给定一棵二叉树前序遍历中序遍历的结果,求其后遍历的方法比较简单,可以通过递归的方式解决。首先,我们需要了解一下前序遍历中序遍历和后遍历的概念: 1.前序遍历:先访问根节点,然后按照左子树、右子树的顺序遍历整个二叉树。 2.中序遍历:按照左子树、根节点、右子树的顺序遍历整个二叉树。 3.后遍历:按照左子树、右子树、根节点的顺序遍历整个二叉树。 在给定前序遍历中序遍历的情况下,我们可以根据前序遍历数组第一个元素为根节点,然后再根据中序遍历数组找到根节点的位置,从而确定左右子树的大小,然后递归构建左右子树。最后,将左子树的后遍历、右子树的后遍历和根节点拼接在一起即为所求的后遍历。 下面是具体的步骤: 1.根据前序遍历数组,找到根节点,并创建一个新的节点。 2.在中序遍历数组找到根节点的位置,确定左子树和右子树的大小。 3.递归构建左子树和右子树。 4.将左子树的后遍历、右子树的后遍历和根节点一起拼接起来,得到整棵树的后遍历。 下面是具体的实现代码: ```python class TreeNode: def __init__(self, val=0, left=None, right=None): self.val = val self.left = left self.right = right def buildTree(preorder, inorder): if not preorder: return None root = TreeNode(preorder[0]) idx = inorder.index(preorder[0]) root.left = buildTree(preorder[1:idx+1], inorder[:idx]) root.right = buildTree(preorder[idx+1:], inorder[idx+1:]) return root def postorderTraversal(root): if not root: return [] else: return postorderTraversal(root.left) + postorderTraversal(root.right) + [root.val] preorder = [1, 2, 4, 5, 3, 6, 7] inorder = [4, 2, 5, 1, 6, 3, 7] root = buildTree(preorder, inorder) print(postorderTraversal(root)) #[4, 5, 2, 6, 7, 3, 1] ``` 运行结果为:[4, 5, 2, 6, 7, 3, 1],与我们预期的后遍历结果一致。 总结:给定一棵二叉树前序遍历中序遍历的结果,求其后遍历的方法,可以通过递归的方式实现。通过前序遍历数组找到根节点,再根据中序遍历数组确定左右子树大小,递归构建左右子树,并将左子树的后遍历、右子树的后遍历和根节点拼接在一起,得到整棵树的后遍历。 ### 回答3: 二叉树前序遍历顺序为根节点、左子树、右子树;中序遍历顺序为左子树、根节点、右子树;后遍历顺序为左子树、右子树、根节点。因此,我们可以通过前序遍历中序遍历来构建出二叉树,再通过后遍历输出二叉树遍历结果。 具体步骤如下: 1. 按照前序遍历顺序得到根节点的值; 2. 在中序遍历序列找到根节点的位置,从而确定左右子树的中序遍历; 3. 根据左右子树的中序遍历确定前序遍历左右子树的范围; 4. 递归构建左右子树; 5. 输出后遍历结果。 代码如下: ``` def buildTree(preorder, inorder): if not inorder: return None root_val = preorder.pop(0) root = TreeNode(root_val) index = inorder.index(root_val) root.left = buildTree(preorder, inorder[:index]) root.right = buildTree(preorder, inorder[index+1:]) return root def postorderTraversal(root): res = [] if not root: return res res.extend(postorderTraversal(root.left)) res.extend(postorderTraversal(root.right)) res.append(root.val) return res preorder = [1, 2, 4, 5, 3, 6, 7] inorder = [4, 2, 5, 1, 6, 3, 7] root = buildTree(preorder, inorder) print(postorderTraversal(root)) # 输出:[4, 5, 2, 6, 7, 3, 1] ``` 上述代码的 `buildTree` 函数用于根据前序遍历中序遍历构建二叉树,返回根节点;`postorderTraversal` 函数用于输出二叉树的后遍历结果,采用先递归遍历左右子树的方法。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值