遍历是对树的一种最基本的运算,所谓遍历二叉树,就是按一定的规则和顺序走遍二叉树的所有结点,使每一个结点都被访问一次,而且只被访问一次。由于二叉树是非线性结构,因此,树的遍历实质上是将二叉树的各个结点转换成为一个线性序列来表示。
设L、D、R分别表示遍历左子树、访问根结点和遍历右子树, 则对一棵二叉树的遍历有三种情况:DLR(称为先根次序遍历),LDR(称为中根次序遍历),LRD (称为后根次序遍历)。
首先访问根,再先序遍历左(右)子树,最后先序遍历右(左)子树,C语言代码如下:
1 2 3 4 5 6 7 |
|
首先中序遍历左(右)子树,再访问根,最后中序遍历右(左)子树,C语言代码如下
1 2 3 4 5 6 7 8 |
|
首先后序遍历左(右)子树,再后序遍历右(左)子树,最后访问根,C语言代码如下
1 2 3 4 5 6 7 |
|
即按照层次访问,通常用队列来做。访问根,访问子女,再访问子女的子女(越往后的层次越低)(两个子女的级别相同)
我们首先找到根结点:一定是先序遍历序列的第一个元素:1
然后,在中序序列寻找根,把中序序列分为两个序列左子树4,7,2和右子树5,3,8,6
把先序序列也分为两个: 左子树2,4,7和右子树3,5,6,8
对左右重复同样的过程:
先看左子树:先序序列4,7,2,说明4一定是左子树的根
把2,4,7分为2和7两个序列,再重复过程,左边确定完毕。
右子树同样:中序序列为5,3,8,6,先序序列为:3,5,6,8
取先序头,3.一定是根
把中序序列分为 5和8,6两个序列
对应的先序序列为 5和6,8两个序列
然后确定了5是3的左孩子
对于先序序列6,8和中序序列8,6
还是先取先序的头,6
现在只有8,中序序列8在左边,是左孩子。
结束。
我们总结一下这种方法的过程:
1、根据先序序列确定当前树的根(第一个元素)。
2、在中序序列中找到根,并以根为分界分为两个序列。
3、这样,确定了左子树元素个数,把先序序列也分为两个。
对左右子树(对应的序列)重复相同的过程。
我们把思路用代码实现:
# -*- coding:utf-8 -*-
# class TreeNode:
# def __init__(self, x):
# self.val = x
# self.left = None
# self.right = None
class Solution:
# 返回构造的TreeNode根节点
def reConstructBinaryTree(self, pre, tin):
# write code here/
#pre-先序数组 tin->中序数组
if len(pre) == 0:
return None
root = TreeNode(pre[0])//第一个元素为根
pos = tin.index(pre[0])//划分左右子树
root.left = self.reConstructBinaryTree( pre[1:1+pos], tin[:pos])
root.right = self.reConstructBinaryTree( pre[pos+1:], tin[pos+1:])
return root
思路是类似的,只是我们确定根的时候,取后序序列的最后一个元素即可。
我们直白的表述一下,前序是中左右,后序是左右中。
所以,我们凭先序和后序序列其实是无法判断根的孩子到底是左孩子还是右孩子。
比如先序序列1,5,后序序列是5,1
我们只知道1是这棵树的根,但是我们不知道5是1的左孩子还是右孩子。
我们的中序序列是左中右,才可以明确的划分出左右子树,而先序后序不可以。
综上,只有,只含叶子结点或者同时有左右孩子的结点的树,才可以被先序序列后序序列确定唯一一棵树。
最后不断划分先序和后序序列完成重建。