欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java語言實(shí)現(xiàn)非遞歸實(shí)現(xiàn)樹的前中后序遍歷總結(jié)

 更新時(shí)間:2019年01月03日 14:35:04   作者:sdr_zd  
今天小編就為大家分享一篇關(guān)于Java語言實(shí)現(xiàn)非遞歸實(shí)現(xiàn)樹的前中后序遍歷總結(jié),小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧

前言

三種遍歷的遞歸寫法都很好寫,所以總結(jié)一下非遞歸寫法。

先貼一張圖復(fù)習(xí)一下三種遍歷方式就進(jìn)入正文啦~

【注:本文所有代碼實(shí)現(xiàn)中樹的結(jié)點(diǎn)定義如下:

public class Node {
  int val;
  Node left;
  Node right;
  Node parent;
  Node() {}
  Node(int val) {
    this.val = val;
  }
}

1.前序遍歷

實(shí)現(xiàn)思路:

前序遍歷的順序是:根結(jié)點(diǎn) -> 左孩子 -> 右孩子

借助一個(gè)棧結(jié)構(gòu)先將根結(jié)點(diǎn)壓入棧,然后循環(huán)每次取出棧頂元素,直到棧為空。如果當(dāng)前結(jié)點(diǎn)有右孩子就先將右孩子壓入棧中,如果當(dāng)前結(jié)點(diǎn)有左孩子就將左孩子壓入棧中,這里注意順序,因?yàn)闂5慕Y(jié)構(gòu)是先進(jìn)后出的,為了保證先遍歷到左孩子就應(yīng)該后壓左孩子~

代碼實(shí)現(xiàn):

【代碼使用直接輸出的方式打印中序遍歷結(jié)果,也可以放入一個(gè)list中存儲(chǔ)~】

public void preOrder(Node head) {
    System.out.println("pre-order:");
    if(head != null) {
      Stack<Node> stack = new Stack<>();
      stack.push(head);
      while(!stack.isEmpty()) {
        head = stack.pop();
        System.out.print(head.val + " ");
        if (head.right != null)
          stack.push(head.right);
        if (head.left != null)
          stack.push(head.left);
      }
    }
  }

2.中序遍歷

實(shí)現(xiàn)思路:

中序遍歷的順序:左孩子 -> 根結(jié)點(diǎn) -> 右孩子

借助棧結(jié)構(gòu),將當(dāng)前結(jié)點(diǎn)的左孩子依次入棧直到?jīng)]有左孩子了就彈出棧頂元素并打印,如果彈出的結(jié)點(diǎn)有右孩子就將右孩子的左邊界依次入棧,循環(huán)…

比如文章開始的那個(gè)圖中,先將A,B,D依次入棧,此時(shí)棧頂元素是D,彈出并打印,D結(jié)點(diǎn)有右孩子,將D的右孩子的左邊界依次入棧,壓入結(jié)點(diǎn)G,結(jié)點(diǎn)G沒有左孩子所以彈出并打印G,此時(shí)棧頂元素是B,循環(huán)…直到棧中為空且當(dāng)前結(jié)點(diǎn)為空時(shí)遍歷結(jié)束。

代碼實(shí)現(xiàn):

public static void inOrderTraverse(Node head) {
    System.out.println("in-order:");
    if(head != null) {
      Stack<Node> stack = new Stack<>();
      while(!stack.isEmpty() || head != null) {
        if(head != null) {
          // 當(dāng)前節(jié)點(diǎn)不為空, 將自己壓進(jìn)棧并將自己的左孩子作為當(dāng)前節(jié)點(diǎn)(壓入左邊界)
          stack.push(head);
          head = head.left;
        }else {
          // 當(dāng)前節(jié)點(diǎn)為空(沒有左孩子了), 將棧頂元素彈出作為當(dāng)前節(jié)點(diǎn), 并將當(dāng)前節(jié)點(diǎn)的右孩子壓進(jìn)棧
          head = stack.pop();
          System.out.print(head.val + " ");
          head = head.right;
        }
      }
    }
  }

3.后序遍歷

實(shí)現(xiàn)思路:

后序遍歷的順序:左孩子 -> 右孩子 -> 根結(jié)點(diǎn)

仔細(xì)想一下,打印每個(gè)結(jié)點(diǎn)需要訪問根結(jié)點(diǎn)三次,先從根結(jié)點(diǎn)開始找到左孩子,返回再找到右孩子,再返回到根結(jié)點(diǎn),需要訪問根結(jié)點(diǎn)三次,直接按照當(dāng)前順序進(jìn)行遍歷不知如何下手,那么我們可以換一個(gè)角度去考慮。

棧結(jié)構(gòu)是先進(jìn)后出的,那我們不妨借助一個(gè)棧先存儲(chǔ) 根結(jié)點(diǎn) -> 右孩子 -> 左孩子 的結(jié)果,再將其依次彈出就是后序遍歷的順序了。

那實(shí)現(xiàn) 根結(jié)點(diǎn) -> 右孩子 -> 左孩子 的方法就很簡單了,回憶一下剛才說的前序遍歷的方式,前序遍歷是先要左孩子,就后壓入左孩子,反之,將前序遍歷的邏輯改寫為:彈出每個(gè)棧頂結(jié)點(diǎn)作為當(dāng)前結(jié)點(diǎn)并存儲(chǔ)到一個(gè)輔助棧中,如果當(dāng)前結(jié)點(diǎn)有左孩子就先壓入左孩子,如果有右孩子就后壓入右孩子,每次取棧頂彈出到輔助棧中。最后將得到的輔助棧中元素依次彈出得到的就是后序遍歷的結(jié)果。

代碼實(shí)現(xiàn):

public static void posOrderTraverse(Node head) {
    System.out.println("pos-order");
    if(head != null) {
      Stack<Node> stack1 = new Stack<>();
      Stack<Node> stack2 = new Stack<>();   // 輔助棧,存儲(chǔ) 根 -> 右 -> 左 的結(jié)果
      stack1.push(head);
      while(!stack1.isEmpty()) {
        head = stack1.pop();
        stack2.push(head);
        // 有左孩子就先壓入左孩子
        if(head.left != null)
          stack1.push(head.left);
        // 有右孩子就后壓入右孩子
        if(head.right != null)
          stack1.push(head.right);
      }
      // 逆序打印 根 -> 右 -> 左 的結(jié)果,就是后序遍歷的結(jié)果
      while(!stack2.isEmpty())
        System.out.print(stack2.pop().val + " ");
    }
  }

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接

相關(guān)文章

  • SpringBoot實(shí)現(xiàn)對(duì)超大文件進(jìn)行異步壓縮下載的使用示例

    SpringBoot實(shí)現(xiàn)對(duì)超大文件進(jìn)行異步壓縮下載的使用示例

    在Web應(yīng)用中,文件下載功能是一個(gè)常見的需求,本文介紹了SpringBoot實(shí)現(xiàn)對(duì)超大文件進(jìn)行異步壓縮下載的使用示例,具有一定的參考價(jià)值,感興趣的可以了解一下,
    2023-09-09
  • 如何利用Java遞歸解決“九連環(huán)”公式

    如何利用Java遞歸解決“九連環(huán)”公式

    這篇文章主要給大家介紹了關(guān)于如何利用Java遞歸解決“九連環(huán)”公式的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • Maven中錯(cuò)誤使用parent.relativePath導(dǎo)致構(gòu)建失敗問題

    Maven中錯(cuò)誤使用parent.relativePath導(dǎo)致構(gòu)建失敗問題

    這篇文章主要介紹了Maven中錯(cuò)誤使用parent.relativePath導(dǎo)致構(gòu)建失敗問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • Gradle環(huán)境下導(dǎo)出Swagger為PDF的步驟詳解

    Gradle環(huán)境下導(dǎo)出Swagger為PDF的步驟詳解

    這篇文章主要介紹了Gradle環(huán)境下導(dǎo)出Swagger為PDF的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-06-06
  • Java創(chuàng)建對(duì)象的幾種方法

    Java創(chuàng)建對(duì)象的幾種方法

    這篇文章主要為大家詳細(xì)介紹了Java創(chuàng)建對(duì)象的幾種方法,使用new創(chuàng)建、使用object.clone()創(chuàng)建、使用反序列化創(chuàng)建等,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • Java線程的調(diào)度與優(yōu)先級(jí)詳解

    Java線程的調(diào)度與優(yōu)先級(jí)詳解

    這篇文章主要為大家詳細(xì)介紹了Java線程的調(diào)度與優(yōu)先級(jí),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-03-03
  • 解決@CachePut設(shè)置的key值無法與@CacheValue的值匹配問題

    解決@CachePut設(shè)置的key值無法與@CacheValue的值匹配問題

    這篇文章主要介紹了解決@CachePut設(shè)置的key的值無法與@CacheValue的值匹配問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • Resttemplate上傳文件500異常的原因及解決方法

    Resttemplate上傳文件500異常的原因及解決方法

    使用 Resttemplate 調(diào)用 DMS 文件服務(wù)器 Http 接口,出現(xiàn) 500 異常報(bào)錯(cuò),所以本文給大家介紹了Resttemplate上傳文件500異常的原因及解決方法,需要的朋友可以參考下
    2024-08-08
  • JAVA線程同步實(shí)例教程

    JAVA線程同步實(shí)例教程

    這篇文章主要介紹了JAVA線程同步實(shí)例教程,在Java程序設(shè)計(jì)中有著非常廣泛的應(yīng)用,需要的朋友可以參考下
    2014-08-08
  • eclipse maven maven-archetype-webapp 創(chuàng)建失敗問題解決

    eclipse maven maven-archetype-webapp 創(chuàng)建失敗問題解決

    這篇文章主要介紹了eclipse maven maven-archetype-webapp 創(chuàng)建失敗問題解決的相關(guān)資料,需要的朋友可以參考下
    2016-12-12

最新評(píng)論