zoukankan      html  css  js  c++  java
  • 二叉树的遍历(C++非递归实现)

    文章真是经典!

    原文:http://www.cnblogs.com/microgrape/archive/2011/05/11/2043799.html

    其实二叉树的遍历,这么经典的东西,我自己一直都不明白。递归算法那么简洁,让人神魂颠倒;非递归算法就那么不好理解。。。最近使劲想了一想,似乎有一些眉目了,于是记录下来。

    【思想】

    (1)递归思想。虽然非递归算法没有直接使用函数递归,但是使用了栈,所以实际上仍然是递归的思想。但是它递归的是那么朦胧,让人无法捉摸。可以从以下几个方面思考:首先,三个非递归遍历的算法开始都是向左走到树的最左下方的节点,并把路上遇到的节点入栈,这其实就是递归中的前一半过程。如下图所示:


    图中上半部分表示了一个递归函数的层层深入的栈结构,下面则是模拟了将二叉树的节点入栈的过程。以此类推,从最左下的节点向上的过程就是退栈的过程,也就是图上半部分的右边。

    (2)算法中cur节点的实际意义。在下面的代码中,可以看到三个算法(前、中、后序)中都有一个cur节点。这个节点表示什么含义呢?就是表示发现了一个继续入栈的机会,也就是从这个节点开始,又可以重复向左下走到底。所以中序遍历和后序遍历的开始都有一句:

    while( ! s.empty() || cur != NULL )

    那么当cur为空,也就是没有可以继续入栈的机会的时候,那就得靠栈中存着的余货了。

    对中序遍历来说,访问完当前节点之后,如果它没有右孩子,那就没有继续入栈的机会了,只能退栈。

    对后序遍历来说,检查当前节点时,如果当前节点没有右孩子或者它的右孩子刚刚被访问,那么也别无选择,只能退栈。

    这也就是它们的代码中把相应的cur置为NULL的原因。

    【代码】

    #include <iostream>
    #include <stack>
    #include <vector>
    #include <iomanip>
    #include <queue>
    
    using namespace std;
    
    struct Node
    {
          Node( const Node *t ) : data( t->data ), lChild( NULL ), rChild( NULL ) {}
          Node( int d ) : data( d ), lChild( NULL ), rChild( NULL ) {}
    
        struct Node* lChild;
        struct Node* rChild;
        int data;
    };
    
    void PreOrder( Node *root, void (*visit)(Node*) )  //ps:二叉树的先序遍历与二叉树的深度优先遍历什么关系呢?
    {
    if( root == NULL ) return;
    stack<Node*> s;
    Node *cur;
    s.push( root );
    
    while( !s.empty() ) 
    {
       cur = s.top();   visit( cur );     s.pop();
      if( cur->rChild != NULL )   s.push( cur->rChild );
      if( cur->lChild != NULL )   s.push( cur->lChild );
    }
    }
    
    void InOrder( Node *root, void (*visit)(Node*) )
    {
    stack<Node*> s;
    Node *cur = root;
    
    while( ! s.empty() || cur != NULL )    
    {
      while( cur != NULL )
       {    s.push( cur ); cur = cur->lChild;    }
    
       cur = s.top(); visit( cur ); s.pop();
    
      if( cur->rChild == NULL )
        cur = NULL; // no current node. must pop.
      else
        cur = cur->rChild;  
    }
    }
    
    void PostOrder( Node *root, void (*visit)(Node*) )
    {
    stack<Node*> s;
    Node *cur = root;
    Node *visited = NULL;
    
    while( ! s.empty() || cur != NULL )
    {
      while( cur != NULL )
       {     s.push( cur ); cur = cur->lChild; }
    
       cur = s.top();  // check but no visit.
    
      if( cur->rChild == visited || cur->rChild == NULL )
       {
        visit( cur ); s.pop();
        visited = cur;
        cur = NULL; // no current node, must pop.
       }
      else
        cur = cur->rChild;
    }
    }
    
    /***************************************
    * 层序遍历:跟先序遍历很像。是二叉树的广度优先搜索。
    * 所以要用队列实现。
    ***************************************/
    void LevelOrderTraverse(Node* t, void (*visit)(Node* ) )
    {
        if(NULL == t)
            return;
          queue<Node*> q;
          Node* e;
          q.push(t);
        while(!q.empty())
          {
              visit(e = q.front());
              q.pop();
            if(NULL != e->lChild) q.push(e->lChild);
            if(NULL != e->rChild) q.push(e->rChild);
          }
    }
    
    //插入节点的函数
    void insert(Node *&root, Node *t)
    {
        if( root == NULL )
              root = new Node( t );
        else if ( t->data > root->data )
              insert( root->rChild, t );
        else if ( t->data < root->data )
              insert( root->lChild, t );
        else
              ;
    }
    
    //使用数组创建二叉树
    Node* createTree( vector<int>& a)
    {
        if( 0 == a.size() )
            return NULL;
    
          Node *root = new Node(a[0]);
          vector<int>::const_iterator iter = a.begin();
          iter++;
        for ( ; iter != a.end() ; ++iter )
          {
              Node *t = new Node( *iter );
              insert( root, t );
          }
        return root;
    }
    
    void visit(Node *t)
    {
          cout<<setw(3)<<t->data<<" ";
    }

    下面是主函数:

    int main()
    {
         vector<int> a;
        for (int i = 0 ; i < 10 ; ++i )
         {
            int temp = rand()%31;//随便取了几个随机数
             a.push_back(temp);
         }
    
         Node* root = createTree( a );
    
         PreOrderTraverse(root, visit);
         cout<<endl;
         InOrderTraverse(root, visit);
         cout<<endl;
         PostOrderTraverse(root, visit);
         cout<<endl;
    }
  • 相关阅读:
    解压版(绿色版)Tomcat配置
    安装、设置与启动MySql绿色版的方法
    JDBC连接MySQL出现的问题
    可变长数组在GCC编译器中的实现
    关于取余
    java中类和接口的一点整理
    orgmode 写cnblogs 博文(一些问题的解决)
    This file is about changes in Emacs version 24.
    autocomplete插件使用中遇到的问题及原因
    junit&jmockit工具熟悉记录
  • 原文地址:https://www.cnblogs.com/theCambrian/p/3624214.html
Copyright © 2011-2022 走看看