亲宝软件园·资讯

展开

【树01】对二叉树前序/中序/后序遍历算法的一些思考

山斗 人气:1

二叉树的前序、中序、后序遍历

  

 

 

   每个节点会被经过3次,前序、中序、后序的区别在于:在哪一次经过该节点时对其进行访问。

2. 递归实现

traverseRecursive(BiTrNode<T>* node):

    basecase:
        if(node == nullptr) return;

    general:
        1 print(node->data);
        2 traverseRecursive(node->lchild);
        3 traverseRecursive(node->rchild);

        前序:123
        中序:213
        后序:132

3. 非递归实现

  首先需要明确的是,所谓的递归实现其实是在模拟递归栈,所以使用的基本数据结构是stack。

  这里如何设计节点的入栈策略是关键,因为执行的时候无非就是依次从stack中取出栈顶元素进行处理,所以stack中节点的入栈顺序就是访问顺序。

  那既然我们已经知道了,每个节点都会被经过3次,所谓前序、中序、后序的区别无非就是第几次经过时对其进行操作(这里我们把对节点的操作简化成print node->data)。

 

初步想法

  给每一个节点增加一个字段(lookTimes)表示这是第几次经过该节点。

  模拟一下3次经过的入栈顺序:①第一次经过该节点 -> ②遍历该节点的左孩子 -> ③回到该节点(第二次经过)-> ④遍历该节点的右孩子 -> ⑤回到该节点(第三次经过)。

  init:所有节点的lookTimes都设为0,将根节点入栈。

  从stack中取出栈顶元素表示经过一次该节点(lookTimes++),然后看lookTimes进行相应的操作:

  • lookTimes = 1:第一次访问该节点,接下来要访问该节点的左子树,然后再回到该节点,再所以应该依次:push(左孩子),push(自己)
  • lookTimes = 2:第二次访问该节点,接下来要访问该节点的右子树,然后再回到该节点,所以应该依次:push(右孩子),push(自己)
  • lookTimes = 3:第三次访问该节点,此时对该节点的3次访问就结束了,所以就没有什么额外操作等着取下一个节点就行了。

  模拟好3次经过之后,要实现前序/中序/后序遍历只需在每一次push前print(node->data)就可以了。比如,如果是前序遍历就在p->lookTimes == 1的if语句块的第一行加上print(node->data)就可以了。

 

 

 

  伪码如下:

  

//version1.0
while(stack is not empty){
        p = stack.pop();
        (p->lookTimes)++;

        if(p->lookTimes == 1){
            //print(p->data); 前序遍历
            stack.push(p);
            if(p->lchild)
                stack.push(p->lchild);
        }else if(p->lookTimes == 2){
            //print(p->data); 中序遍历
            stack.push(p);
            if(p->lchild)
                stack.push(p->rchild);
        }else if(p->lookTimes == 3){
            //print(p->data); 后序遍历
        }         
    }

 

优化版本

  上面是一种比较通用的想法,但是在考虑到具体的某一种遍历方式时,比如前序遍历:既然在lookTimes == 1时就已经对该节点进行访问了,那么后面的两次回到该节点显然是没有必要的。

  因此根据每一种遍历方式的特点进行优化:

  1. 前序

  入栈顺序:①第一次经过该节点 -> ②遍历该节点的左孩子 -> ③遍历该节点的右孩子。而且这样的话每个节点只会经过一次,lookTimes字段也就没有必要了。

//前序遍历_version2.0:without lookTimes    
while(stack is not empty){
        p = stack.pop();
        print(p->data);

        if(p->lchild)
                stack.push(p->lchild);

        if(p->rchild)
                stack.push(p->rchild);         
 }        

  2. 中序

  入栈顺序:①第一次经过该节点 -> ②遍历该节点的左孩子 -> ③回到该节点(第二次经过)-> ④遍历该节点的右孩子。

//中序遍历_version2.0
while(stack is not empty){
        p = stack.pop();
        (p->lookTimes)++;

        if(p->lookTimes == 1){
            stack.push(p);
            if(p->lchild)
                stack.push(p->lchild);
        }else if(p->lookTimes == 2){
            print(p->data);
            if(p->lchild)
                stack.push(p->rchild);
        }   
    }

  3. 后序

  后序因为在第三次经过时才会访问,所以不能再优化了。

 

加载全部内容

相关教程
猜你喜欢
用户评论