程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> 二叉樹的遍歷的迭代和遞歸實現方式

二叉樹的遍歷的迭代和遞歸實現方式

編輯:C++入門知識

二叉樹是一種非常重要的數據結構,很多其它數據結構都是基於二叉樹的基礎演變而來的。對於二叉樹,有前序、中序以及後序三種遍歷方法。因為樹的定義本身就是遞歸定義,因此采用遞歸的方法去實現樹的三種遍歷不僅容易理解而且代碼很簡潔。而對於樹的遍歷若采用非遞歸的方法,就要采用棧去模擬實現。在三種遍歷中,前序和中序遍歷的非遞歸算法都很容易實現,非遞歸後序遍歷實現起來相對來說要難一點。

一.前序遍歷

   前序遍歷按照“根結點-左孩子-右孩子”的順序進行訪問。

   1.遞歸實現


void preOrder1(BinTree *root)     //遞歸前序遍歷
{
    if(root!=NULL)
    {
        cout<<root->data<<" ";
        preOrder1(root->lchild);
        preOrder1(root->rchild);
    }
}
   2.非遞歸實現

    根據前序遍歷訪問的順序,優先訪問根結點,然後再分別訪問左孩子和右孩子。即對於任一結點,其可看做是根結點,因此可以直接訪問,訪問完之後,若其左孩子不為空,按相同規則訪問它的左子樹;當訪問其左子樹時,再訪問它的右子樹。因此其處理過程如下:

     對於任一結點P:

     1)訪問結點P,並將結點P入棧;

     2)判斷結點P的左孩子是否為空,若為空,則取棧頂結點並進行出棧操作,並將棧頂結點的右孩子置為當前的結點P,循環至1);若不為空,則將P的左孩子置為當前的結點P;

     3)直到P為NULL並且棧為空,則遍歷結束。


void preOrder2(BinTree *root)     //非遞歸前序遍歷
{
    stack<BinTree*> s;
    BinTree *p=root;
    while(p!=NULL||!s.empty())
    {
        while(p!=NULL)
        {
            cout<<p->data<<" ";
            s.push(p);
            p=p->lchild;
        }
        if(!s.empty())
        {
            p=s.top();
            s.pop();
            p=p->rchild;
        }
    }
}
二.中序遍歷

    中序遍歷按照“左孩子-根結點-右孩子”的順序進行訪問。

    1.遞歸實現


void inOrder1(BinTree *root)      //遞歸中序遍歷
{
    if(root!=NULL)
    {
        inOrder1(root->lchild);
        cout<<root->data<<" ";
        inOrder1(root->rchild);
    }
}
   2.非遞歸實現

    根據中序遍歷的順序,對於任一結點,優先訪問其左孩子,而左孩子結點又可以看做一根結點,然後繼續訪問其左孩子結點,直到遇到左孩子結點為空的結點才進行訪問,然後按相同的規則訪問其右子樹。因此其處理過程如下:

   對於任一結點P,

  1)若其左孩子不為空,則將P入棧並將P的左孩子置為當前的P,然後對當前結點P再進行相同的處理;

  2)若其左孩子為空,則取棧頂元素並進行出棧操作,訪問該棧頂結點,然後將當前的P置為棧頂結點的右孩子;

  3)直到P為NULL並且棧為空則遍歷結束


void inOrder2(BinTree *root)      //非遞歸中序遍歷
{
    stack<BinTree*> s;
    BinTree *p=root;
    while(p!=NULL||!s.empty())
    {
        while(p!=NULL)
        {
            s.push(p);
            p=p->lchild;
        }
        if(!s.empty())
        {
            p=s.top();
            cout<<p->data<<" ";
            s.pop();
            p=p->rchild;
        }
    }   
}
  三.後序遍歷

      後序遍歷按照“左孩子-右孩子-根結點”的順序進行訪問。

      1.遞歸實現


void postOrder1(BinTree *root)    //遞歸後序遍歷
{
    if(root!=NULL)
    {
        postOrder1(root->lchild);
        postOrder1(root->rchild);
        cout<<root->data<<" ";
    }   
}
    2.迭代實現
先找到最左邊的葉子並把路上遇到的節點一次進棧,然後彈出棧頂的元素(該元素為最左邊的葉子)並判斷(1)它有沒有右節點;
(2)右節點是否被訪問過;如果有右節點同時沒被訪問過,則先壓入剛才彈出的元素,然後壓入它的右子樹,否則訪問該節點並設置pre為該節點。
void postOrder2(BinTree *root) //非遞歸後序遍歷
{
    stack<BinTree*> s;
    BinTree *p=root;
    BinTree *pre=NULL;
    BinTree *top=NULL;
    while(p!=NULL||!s.empty())
    {
        while(p!=NULL)
        {
            s.push(p);
            p=p->lchild;
        }
        if(!s.empty())
        {
            top=s.top();
            if(top->right!=NULL&&top->right!=pre)
            {
                p=p->right;
            }
            else
            {
                cout<<top->data<<" "
                pre=top;
                s.pop() 
            }
        }
    }   
}

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved