给你一棵二叉树的根节点
root
,返回其节点值的 后序遍历
知识点:栈
栈是一种仅支持在表尾进行插入和删除操作的线性表,这一端被称为栈顶,另一端被称为栈底。元素入栈指的是把新元素放到栈顶元素的上面,使之成为新的栈顶元素;元素出栈指的是从一个栈删除元素又称作出栈或退栈,它是把栈顶元素删除掉,使其相邻的元素成为新的栈顶元素。
思路:
既然二叉树的前序遍历和中序遍历都可以使用栈来代替递归,那后序遍历是否也可以呢?答案是可以的,但是会比前二者复杂一点点。
根据后序遍历“左右根”的顺序,那么后序遍历也与中序遍历类似,要先找到每棵子树的最左端节点:
1 2 3 4 5 |
|
然后我们就要访问该节点了嘛?不不不,如果它还有一个右节点呢?根据“左右根”的原则,我还要先访问右子树。我们只能说它是最左端的节点,它左边为空,但是右边不一定,因此这个节点必须被看成是这棵最小的子树的根。要怎么访问根节点呢?
我们都知道从栈中弹出根节点,一定是左节点已经被访问过了,因为左节点是子问题,访问完了才回到父问题,那么我们还必须要确保右边也已经被访问过了。如果右边为空,那肯定不用去了,如果右边不为空,那我们肯定优先进入右边,此时再将根节点加入栈中,等待右边的子树结束。
1 2 3 4 |
|
不过,当右边被访问了,又回到了根,我们的根怎么知道右边被访问了呢?用一个前序指针pre标记一下,每个根节点只对它的右节点需要标记,而每个右节点自己本身就是一个根节点,因此每次访问根节点的时候,我们可以用pre标记为该节点,回到上一个根节点时,检查一下,如果pre确实是它的右子节点,哦那正好,刚刚已经访问过了,我现在可以安心访问这个根了。
1 2 3 4 5 6 7 |
|
具体做法:
- step 1:开辟一个辅助栈,用于记录要访问的子节点,开辟一个前序指针pre。
- step 2:从根节点开始,每次优先进入每棵的子树的最左边一个节点,我们将其不断加入栈中,用来保存父问题。
- step 3:弹出一个栈元素,看成该子树的根,判断这个根的右边有没有节点或是有没有被访问过,如果没有右节点或是被访问过了,可以访问这个根,并将前序节点标记为这个根。
- step 4:如果没有被访问,那这个根必须入栈,进入右子树继续访问,只有右子树结束了回到这里才能继续访问根。
图解:
代码实现:
List<Integer> res = new ArrayList<>();
if(root == null)
return res;
Stack<TreeNode> s = new Stack<>();
TreeNode pre = null;//已经访问过的节点,用于判断增加根节点时使用
while(root != null || !s.isEmpty()){
while(root != null){
s.push(root);
root = root.left;
}
TreeNode cur = s.pop();
//如果该元素的右边没有或是已经访问过
if(cur.right == null || cur.right == pre){
//访问中间的节点
res.add(cur.val);
pre = cur;
}
else{
s.push(cur);//该节点入栈
root = cur.right;//先访问右节点
}
}
return res;
}