给定一个二叉树,返回它的 后序 遍历。

示例:

输入: [1,null,2,3]
1
\
2
/
3 输出: [3,2,1]

思路:一开始编写二叉树后序遍历的程序,感觉定级为困难有点欠妥,确实,如果用递归的做法来做,和前序中序没有太大的程序上的变动,但是如果用非递归的做法来做,就会发现确实要多了一个判断过程。

(1)递归

    vector<int> a;
vector<int> postorderTraversal(TreeNode* root) {
if(root)
{
postorderTraversal(root->left);
postorderTraversal(root->right);
a.push_back(root->val);
}
return a;
}

(2)非递归

后序遍历的非递归实现是三种遍历方式中最难的一种。因为在后序遍历中,要保证左孩子和右孩子都已被访问并且左孩子在右孩子前访问才能访问根结点,这就为流程的控制带来了难题。下面介绍两种思路。

第一种思路:对于任一结点P,将其入栈,然后沿其左子树一直往下搜索,直到搜索到没有左孩子的结点,此时该结点出现在栈顶,但是此时不能将其出栈并访问,因此其右孩子还没有被访问。所以接下来按照相同的规则对其右子树进行相同的处理,当访问完其右孩子时,该结点又出现在栈顶,此时可以将其出栈并访问。这样就保证了正确的访问顺序。可以看出,在这个过程中,每个结点都两次出现在栈顶,只有在第二次出现在栈顶时,才能访问它。因此需要多设置一个变量标识该结点是否是第一次出现在栈顶。

  struct BTNode
 {
TreeNode* node;
bool isFirst;
BTNode(TreeNode* p): node(p),isFirst(true){}
};
vector<int> postorderTraversal(TreeNode* root) {
vector<int> a;
stack<BTNode*> s;
TreeNode* p=root;
BTNode*temp;
while(p!=NULL || !s.empty())
{
while(p)
{
BTNode* b=new BTNode(p);
s.push(b);
p=p->left;
}
if(!s.empty())
{
temp=s.top();
s.pop();
if(temp->isFirst==true)
{
temp->isFirst=false;
s.push(temp);
p=temp->node->right;
}
else
{
a.push_back(temp->node->val);
p=NULL;
}
}
}
return a;
}

这里多定义了一个结构体,里面包含了一个标志位isFirst,用来判断这个根节点是不是第一次来到栈顶,如果是第一次,isFirst==true,我们需要继续遍历此节点的右子树,并将其置位为false,这个时候要注意,前面已经将此节点pop出来了,要再次将其push到栈中,如果是第二次,就证明它的左右子树都已经遍历过了,所以就直接将此节点的值打印就好了。

第二种思路:要保证根结点在左孩子和右孩子访问之后才能访问,因此对于任一结点P,先将其入栈。如果P不存在左孩子和右孩子,则可以直接访问它;或者P存在左孩子或者右孩子,但是其左孩子和右孩子都已被访问过了,则同样可以直接访问该结点。若非上述两种情况,则将P的右孩子和左孩子依次入栈,这样就保证了每次取栈顶元素的时候,左孩子在右孩子前面被访问,左孩子和右孩子都在根结点前面被访问。

 vector<int> postorderTraversal(TreeNode* root) {
stack<TreeNode*> s;
TreeNode *cur=NULL; //当前结点
TreeNode *pre=NULL; //前一次访问的结点
vector<int> a;
s.push(root);
while(root && !s.empty())
{
cur=s.top();
if((cur->left==NULL&&cur->right==NULL)||(pre!=NULL &&(pre==cur->left||pre==cur->right)))
{
a.push_back(cur->val); //如果当前结点没有孩子结点或者孩子节点都已被访问过
s.pop();
pre=cur;
}
else
{
if(cur->right!=NULL)
s.push(cur->right);
if(cur->left!=NULL)
s.push(cur->left);
}
}
return a;
}

这个思路很好,目的也是要保证遍历过程的正确性,多使用了一个指针pre来存储前一次访问的节点,这样就可以判断此节点的右子树有没有被访问过。判断条件中

(pre!=NULL &&(pre==cur->left||pre==cur->right))

很难理解,一开始会想,这里的pre==cur->left是错误的,因为如果我前一个访问的是该节点的左孩子,那就可以直接访问该节点吗,怎么可能呢。仔细想,就是这样的。因为这种情况只可能出现在,该节点没有右孩子,所以上一个访问完左孩子,直接就可以访问该节点。如果有右孩子在,上一个访问的节点不可能是左孩子,因为右孩子是在此节点之后打入栈中的,会更早的出现在栈顶。

												

最新文章

  1. Spring学习总结(一)——Spring实现IoC的多种方式
  2. SQL 性能调优日常积累【转】
  3. myeclipse中java文件中文注释乱码问题
  4. 编写第一个java程序
  5. Spark及其应用场景初探
  6. 《Maven_孔浩》Maven命令
  7. Ipad亚麻布纹背景-最终效果_学习教程
  8. Python魔法方法详解
  9. cf1073D Berland Fair (二分答案+树状数组)
  10. HDU contest808 ACM多校第7场 Problem - 1008: Traffic Network in Numazu
  11. Linux之路,起步虽晚,迈步才会成功(2013.08.09)
  12. es6 - 函数 扩展
  13. Win10系列:VC++数据绑定
  14. jquery双击事件会触发单击事件
  15. 2018.09.05 bzoj2726: [SDOI2012]任务安排(斜率优化dp+二分)
  16. linux对文件赋权限的命令chmod的详细说明
  17. MT【117】立体几何里的一道分类讨论题
  18. 安装openldap
  19. leetcode第一刷_ Flatten Binary Tree to Linked List
  20. 【随笔】使用apt-spy来更新你的debian源

热门文章

  1. [Usaco2008 Mar]River Crossing渡河问题
  2. CMU数据库(15-445)实验2-b+树索引实现(上)
  3. Py数据类型—整形与字符串
  4. Wi-Fi IoT套件连PCF8563实现电子钟功能
  5. 1、进程管理常用命令和进程ID
  6. KMP算法 字符串匹配(看猫片)
  7. 洛谷P4317
  8. Linux 文件搜索神器 find 实战详解,建议收藏!
  9. Core3.1 微信v3 JSAPI支付 退款
  10. JS小整理