实现二叉树的前序(preorder)、中序(inorder)、后序(postorder)遍历有两个常用的方法:一是递归(recursive),二是使用栈实现的迭代版本(stack+iterative)。这两种方法都是O(n)的空间复杂度(递归本身占用stack空间或者用户自定义的stack)。第三种是Morris方法遍历,它的空间复杂度只有O(1)。
复杂度分析:
迭代版,用栈存放遍历的节点,注意是按照访问节点,压入右节点,压入左节点的顺序。
public List<Integer> preorderTraversal(TreeNode root) {
List<Integer> list = new ArrayList<Integer>();
Stack<TreeNode> stack = new Stack<TreeNode>();
stack.push(root);
while (!stack.isEmpty()) {
TreeNode node = stack.pop();
list.add(node.val);
if (node.right != null) stack.push(node.right);
if (node.left != null) stack.push(node.left);
}
return list;
}
迭代版,将节点全部压入栈中,再弹出。
步骤:
public List<Integer> preorderTraversal(TreeNode root) {
List<Integer> list = new ArrayList<Integer>();
if (root == null)
return list;
Stack<TreeNode> stack = new Stack<TreeNode>();
TreeNode p = root;
while (!stack.isEmpty() || p != null) {
while (p != null) {
stack.push(p);
list.add(p.val);
p = p.left;
}
p = stack.pop();
p = p.right;
}
return list;
}
或者
public List<Integer> preorderTraversal(TreeNode root) {
List<Integer> result = new ArrayList<>();
Deque<TreeNode> stack = new ArrayDeque<>();
TreeNode p = root;
while(!stack.isEmpty() || p != null) {
if(p != null) {
stack.push(p);
result.add(p.val);
p = p.left;
} else {
TreeNode node = stack.pop();
p = node.right;
}
}
return result;
}
迭代版,也使用栈进行储存,但是栈里只存放右节点。每当到达一个节点时,右节点不为空先将右节点压入栈中;然后访问左节点,判断左节点是否为 null,如果为 null,并且栈不为空的话,就弹出之前存放的右子节点继续遍历。如果用示意图来表达的话,就是访问指针每次都先往左子节点遍历,直到为null,然后取出其父节点的右子树继续向左子节点遍历,以此类推。
步骤:
代码:
public List<Integer> preorderTraversal(TreeNode node) {
List<Integer> list = new LinkedList<Integer>();
Stack<TreeNode> rights = new Stack<TreeNode>();
while (node != null) {
list.add(node.val);
if (node.right != null) {
rights.push(node.right);
}
node = node.left;
if (node == null && !rights.isEmpty()) {
node = rights.pop();
}
}
return list;
}
Morris Traversal方法遍历二叉树(非递归,不用栈,O(1)空间),它能实现:
Morris Traversal方法可以做到这两点,与递归和栈实现的不同在于该方法只需要O(1)空间,而且同样可以在O(n)时间内完成。具体可以参看(http://www.cnblogs.com/AnnieKim/archive/2013/06/15/morristraversal.html)
主要步骤:
代码:
public void preorderMorrisTraversal(TreeNode root) {
TreeNode cur = root, prev = null;
while (cur != null) {
if (cur.left == null) {
System.out.println(cur.val);
cur = cur.right;
}
else {
prev = cur.left;
while (prev.right != null && prev.right != cur)
prev = prev.right;
if (prev.right == null) {
System.out.println(cur.val);
prev.right = cur;
cur = cur.left;
}
else {
prev.right = null;
cur = cur.right;
}
}
}
}
递归版本
public List<Integer> preorderTraversal(TreeNode node) {
List<Integer> list = new ArrayList<Integer>();
dfs(node, list);
return list;
}
public void dfs(TreeNode node, List<Integer> list) {
if (node == null)
return;
list.add(node.val);
dfs(node.left, list);
dfs(node.right, list);
}
与前序遍历同样是先遍历左子节点再遍历右子节点,不同之处在于:前序遍历的实现方式是在压入栈的时候就访问了这个节点,而中序遍历的实现方式是从栈中弹出节点之后进行访问,保存其顺序。
先遍历当前节点的左子节点,直到没有左子节点,然后pop出当前最后一个左子节点,访问之,再遍历其右子节点的左子节点,以此类推。如果遍历到某个节点为空,那继续pop出下一个节点。
public List<Integer> inorderTraversal(TreeNode root) {
List<Integer> list = new ArrayList<Integer>();
if (root == null)
return list;
Stack<TreeNode> stack = new Stack<TreeNode>();
TreeNode p = root;
while ( !stack.isEmpty() || p != null ) {
while (p != null) {
stack.push(p);
p = p.left;
}
p = stack.pop();
list.add(p.val);
p = p.right;
}
return list;
}
或者
public List<Integer> inorderTraversal(TreeNode root) {
List<Integer> result = new ArrayList<>();
Deque<TreeNode> stack = new ArrayDeque<>();
TreeNode p = root;
while(!stack.isEmpty() || p != null) {
if(p != null) {
stack.push(p);
p = p.left;
} else {
TreeNode node = stack.pop();
result.add(node.val);
p = node.right;
}
}
return result;
}
Morris Traversal中序遍历,非递归。
步骤:
图示:
下图为每一步迭代的结果(从左至右,从上到下),cur代表当前节点,深色节点表示该节点已输出。
代码:
public void inorderMorrisTraversal(TreeNode root) {
TreeNode cur = root, prev = null;
while (cur != null) {
if (cur.left == null) {
System.out.println(cur.val);
cur = cur.right;
}
else {
prev = cur.left;
while (prev.right != null && prev.right != cur)
prev = prev.right;
if (prev.right == null) {
prev.right = cur;
cur = cur.left;
}
else {
prev.right = null;
System.out.println(cur.val);
cur = cur.right;
}
}
}
}
O(1)空间复杂度
O(n)时间复杂度
空间复杂度:O(1),因为只用了两个辅助指针。
时间复杂度:O(n)。证明时间复杂度为O(n),最大的疑惑在于寻找中序遍历下二叉树中所有节点的前驱节点的时间复杂度是多少,即以下两行代码:
while (prev->right != NULL && prev->right != cur)
prev = prev->right;
直觉上,认为它的复杂度是O(nlgn),因为找单个节点的前驱节点与树的高度有关。但事实上,寻找所有节点的前驱节点只需要O(n)时间。n个节点的二叉树中一共有n-1条边,整个过程中每条边最多只走2次,一次是为了定位到某个节点,另一次是为了寻找上面某个节点的前驱节点,如下图所示,其中红色是为了定位到某个节点,黑色线是为了找到前驱节点。所以复杂度为O(n)。
递归版,参数传递结果。
public List<Integer> inorderTraversal(TreeNode node) {
List<Integer> list = new ArrayList<Integer>();
dfs(node, list);
return list;
}
public void dfs(TreeNode node, List<Integer> list) {
if (node == null)
return;
dfs(node.left, list);
list.add(node.val);
dfs(node.right, list);
}
递归版,返回值作为结果。
public List<Integer> inorderTraversal(TreeNode root) {
List<Integer> list = new ArrayList<Integer>();
if (root == null)
return list;
List<Integer> left = inorderTraversal(root.left);
if (left != null) list.addAll(left);
list.add(root.val);
List<Integer> right = inorderTraversal(root.right);
if (right != null) list.addAll(right);
return list;
}
与前序遍历的解法一类似,前序遍历每次先压入右孩子然后再左孩子(访问顺序先左后右),后序遍历正好相反,压入栈时先左后右(访问顺序先右后左)。后序遍历从根节点开始,越早遍历的节点越晚打印,因此需要后入先出的数据结构来保存结果,实际上队列和栈都可以。
public List<Integer> postorderTraversal(TreeNode root) {
LinkedList<Integer> list = new LinkedList<>();
Stack<TreeNode> stack = new Stack<TreeNode>();
stack.push(root);
while (!stack.isEmpty()) {
TreeNode node = stack.pop();
list.addFirst(node.val);
if (node.left != null) stack.push(node.left);
if (node.right != null) stack.push(node.right);
}
return list;
}
与前序遍历的解法二类似。关键在于倒序输出,先访问右孩子再访问左孩子。
public List<Integer> postorderTraversal(TreeNode root) {
LinkedList<Integer> list = new LinkedList<Integer>();
if (root == null)
return list;
Stack<TreeNode> stack = new Stack<TreeNode>();
TreeNode p = root;
while ( !stack.isEmpty() || p != null ) {
while (p != null) {
stack.push(p);
list.addFirst(p.val);
p = p.right;
}
p = stack.pop();
p = p.left;
}
return list;
}
或者
public List<Integer> postorderTraversal(TreeNode root) {
LinkedList<Integer> result = new LinkedList<>();
Deque<TreeNode> stack = new ArrayDeque<>();
TreeNode p = root;
while(!stack.isEmpty() || p != null) {
if(p != null) {
stack.push(p);
result.addFirst(p.val);
p = p.right;
} else {
TreeNode node = stack.pop();
p = node.left;
}
}
return result;
}
关于层序遍历请看每日一恋 - LeetCode 102 & 107. 二叉树的层次遍历
本文是我参考了网上的资料和加上自己的理解总结的,如果有什么不对的地方非常欢迎大家的指正。
参考:
http://www.cnblogs.com/AnnieKim/archive/2013/06/15/morristraversal.html
https://www.yunaitong.cn/binary-tree-traverse.html