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

Java開發(fā)深入分析講解二叉樹的遞歸和非遞歸遍歷方法

 更新時間:2022年05月16日 10:17:50   作者:有什么奇怪!  
樹是一種重要的非線性數(shù)據(jù)結構,直觀地看,它是數(shù)據(jù)元素(在樹中稱為結點)按分支關系組織起來的結構,很象自然界中的樹那樣。樹結構在客觀世界中廣泛存在,如人類社會的族譜和各種社會組織機構都可用樹形象表示,本篇介紹二叉樹的遞歸與非遞歸遍歷的方法

前言

二叉樹的遍歷方法分為前序遍歷,中序遍歷,后續(xù)遍歷,層序遍歷。

1.遞歸遍歷

對于遞歸,就不得不說遞歸三要素:以前序遍歷為例

遞歸入?yún)?shù)和返回值

因為要打印出前序遍歷節(jié)點的數(shù)值,所以參數(shù)里需要傳入List在放節(jié)點的數(shù)值,除了這一點就不需要在處理什么數(shù)據(jù)了也不需要有返回值,所以遞歸函數(shù)返回類型就是void,代碼如下:

public void preorder(TreeNode root, List<Integer> result)

確定終止條件

在遞歸的過程中,如何算是遞歸結束了呢,當然是當前遍歷的節(jié)點是空了,那么本層遞歸就要要結束了,所以如果當前遍歷的這個節(jié)點是空,就直接return

if (root == null) return;

單層循環(huán)邏輯

前序遍歷是中左右的循序,所以在單層遞歸的邏輯,是要先取中節(jié)點的數(shù)值,代碼如下:

result.add(root.val);
preorder(root.left, result);
preorder(root.right, result);
// 前序遍歷·遞歸·LC144_二叉樹的前序遍歷
class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        List<Integer> result = new ArrayList<Integer>();
        preorder(root, result);
        return result;
    }
    public void preorder(TreeNode root, List<Integer> result) {
        if (root == null) {
            return;
        }
        result.add(root.val);//先保存中間節(jié)點
        preorder(root.left, result); //處理左邊節(jié)點
        preorder(root.right, result); //處理右邊節(jié)點
    }
}
// 中序遍歷·遞歸·LC94_二叉樹的中序遍歷
class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        inorder(root, res);
        return res;
    }
    void inorder(TreeNode root, List<Integer> list) {
        if (root == null) {
            return;
        }
        inorder(root.left, list); //先處理左邊節(jié)點
        list.add(root.val);       //保存中間當前的節(jié)點
        inorder(root.right, list);//先處理右邊節(jié)點
    }
}
// 后序遍歷·遞歸·LC145_二叉樹的后序遍歷
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        postorder(root, res);
        return res;
    }
    void postorder(TreeNode root, List<Integer> list) {
        if (root == null) {
            return;
        }
        postorder(root.left, list);  //先處理左邊節(jié)點
        postorder(root.right, list); //再處理右邊節(jié)點
        list.add(root.val);          //保存最后  
    }
}

2.非迭代遍歷

//前序遍歷
class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        Stack<TreeNode> stack = new Stack();
        if (root == null) return res;
        stack.push(root);
        while (!stack.isEmpty()) {
            TreeNode node = stack.pop();
            res.add(node.val);
            if (node.right != null) { //先將右孩子入棧,因為它在最后
                stack.push(node.right);
            }
            if (node.left != null) { //左孩子入棧再出棧
                stack.push(node.left);
            }
        }
        return res;
    }
}
//中序遍歷
class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        if (root == null) return res;
        Stack<TreeNode> stack = new Stack();
        TreeNode cur = root;
        while (cur != null || !stack.isEmpty()) {
            //如果可以,一直往左下探
            if (cur != null) {
                stack.push(cur);
                cur = cur.left;
            } else {
                cur = stack.pop(); //彈出來的肯定是葉子節(jié)點或中間節(jié)點
                res.add(cur.val); //將這個節(jié)點加入list
                cur = cur.right; //查看當前節(jié)點是否有右節(jié)點,如果右,肯定是中間節(jié)點,如果沒有,就是葉子節(jié)點,繼續(xù)彈出就可以
            }
        }
        return res;
    }
}
//后序遍歷
//再來看后序遍歷,先序遍歷是中左右,后續(xù)遍歷是左右中,那么我們只需要調(diào)整一下先序遍歷的代碼順序,就變成中右左的遍歷順序,然后在反轉result數(shù)組,輸出的結果順序就是左右中
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        if (root == null) return res;
        Stack<TreeNode> stack = new Stack();
        stack.push(root);
        while (!stack.isEmpty()) {
            TreeNode node = stack.pop();
            res.add(node.val);
            if (node.left != null) stack.push(node.left); // 相對于前序遍歷,這更改一下入棧順序 (空節(jié)點不入棧)
            if (node.right != null) stack.push(node.right);// 空節(jié)點不入棧 
        }
        Collections.reverse(res); // 將結果反轉之后就是左右中的順序了
        return res;
    }
}

3.二叉樹的統(tǒng)一迭代法

//前序遍歷
class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        List<Integer> result = new LinkedList<>();
        Stack<TreeNode> st = new Stack<>();
        if (root != null) st.push(root);
        while (!st.empty()) {
            TreeNode node = st.peek();
            if (node != null) {
                st.pop(); // 將該節(jié)點彈出,避免重復操作,下面再將右中左節(jié)點添加到棧中
                if (node.right!=null) st.push(node.right);  // 添加右節(jié)點(空節(jié)點不入棧)
                if (node.left!=null) st.push(node.left);    // 添加左節(jié)點(空節(jié)點不入棧)
                st.push(node);                          // 添加中節(jié)點
                st.push(null); // 中節(jié)點訪問過,但是還沒有處理,加入空節(jié)點做為標記。
            } else { // 只有遇到空節(jié)點的時候,才將下一個節(jié)點放進結果集
                st.pop();           // 將空節(jié)點彈出
                node = st.peek();    // 重新取出棧中元素
                st.pop();
                result.add(node.val); // 加入到結果集
            }
        }
        return result;
    }
}
//中序遍歷
class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> result = new LinkedList<>();
        Stack<TreeNode> st = new Stack<>();
        if (root != null) st.push(root);
        while (!st.empty()) {
            TreeNode node = st.peek();
            if (node != null) {
                st.pop(); // 將該節(jié)點彈出,避免重復操作,下面再將右中左節(jié)點添加到棧中
                if (node.right!=null) st.push(node.right);  // 添加右節(jié)點(空節(jié)點不入棧)
                st.push(node);                          // 添加中節(jié)點
                st.push(null); // 中節(jié)點訪問過,但是還沒有處理,加入空節(jié)點做為標記。
                if (node.left!=null) st.push(node.left);    // 添加左節(jié)點(空節(jié)點不入棧)
            } else { // 只有遇到空節(jié)點的時候,才將下一個節(jié)點放進結果集
                st.pop();           // 將空節(jié)點彈出
                node = st.peek();    // 重新取出棧中元素
                st.pop();
                result.add(node.val); // 加入到結果集
            }
        }
        return result;
    }
}
//后序遍歷
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> result = new LinkedList<>();
        Stack<TreeNode> st = new Stack<>();
        if (root != null) st.push(root);
        while (!st.empty()) {
            TreeNode node = st.peek();
            if (node != null) {
                st.pop(); // 將該節(jié)點彈出,避免重復操作,下面再將右中左節(jié)點添加到棧中
                st.push(node);                          // 添加中節(jié)點
                st.push(null); // 中節(jié)點訪問過,但是還沒有處理,加入空節(jié)點做為標記。
                if (node.right!=null) st.push(node.right);  // 添加右節(jié)點(空節(jié)點不入棧)
                if (node.left!=null) st.push(node.left);    // 添加左節(jié)點(空節(jié)點不入棧)         
            } else { // 只有遇到空節(jié)點的時候,才將下一個節(jié)點放進結果集
                st.pop();           // 將空節(jié)點彈出
                node = st.peek();    // 重新取出棧中元素
                st.pop();
                result.add(node.val); // 加入到結果集
            }
        }
        return result;
    }
}

到此這篇關于Java開發(fā)深入分析講解二叉樹的遞歸和非遞歸遍歷方法的文章就介紹到這了,更多相關Java二叉樹的遞歸內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 淺談JackSon的幾種用法

    淺談JackSon的幾種用法

    這篇文章主要介紹了淺談JackSon的幾種用法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-01-01
  • Java多線程實現(xiàn)之Callable詳解

    Java多線程實現(xiàn)之Callable詳解

    這篇文章主要介紹了Java多線程實現(xiàn)之Callable詳解,Callable是一個接口,用于實現(xiàn)多線程,與實現(xiàn)Runnable類似,但是功能更強大,通過實現(xiàn)Callable接口,我們需要重寫call()方法,該方法可以在任務結束后提供一個返回值,需要的朋友可以參考下
    2023-08-08
  • SpringBoot接口如何對參數(shù)進行校驗

    SpringBoot接口如何對參數(shù)進行校驗

    這篇文章主要介紹了SpringBoot接口如何對參數(shù)進行校驗,在以SpringBoot開發(fā)Restful接口時,?對于接口的查詢參數(shù)后臺也是要進行校驗的,同時還需要給出校驗的返回信息放到上文我們統(tǒng)一封裝的結構中
    2022-07-07
  • apollo更改配置刷新@ConfigurationProperties配置類

    apollo更改配置刷新@ConfigurationProperties配置類

    這篇文章主要為大家介紹了apollo更改配置刷新@ConfigurationProperties配置類示例解析,apollo更改配置刷新@ConfigurationProperties配置類
    2023-04-04
  • springboot?vue接口測試前后端樹節(jié)點編輯刪除功能

    springboot?vue接口測試前后端樹節(jié)點編輯刪除功能

    這篇文章主要為大家介紹了springboot?vue接口測試前后端樹節(jié)點編輯刪除功能,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05
  • Java并發(fā)編程之線程中斷

    Java并發(fā)編程之線程中斷

    這篇文章主要介紹了Java并發(fā)編程線程中斷,java線程中斷是一種線程間的協(xié)作模式,通過設置線程的中斷標志并不能直接終止該線程的運行,而是被中斷的線程根據(jù)中斷狀態(tài)自行處理,需要的朋友可以參考一下
    2021-09-09
  • springboot控制層圖片驗證碼生成

    springboot控制層圖片驗證碼生成

    這篇文章主要為大家詳細介紹了springboot控制層圖片驗證碼生成,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-02-02
  • Mybatis應用mysql存儲過程查詢數(shù)據(jù)實例

    Mybatis應用mysql存儲過程查詢數(shù)據(jù)實例

    下面小編就為大家分享一篇Mybatis應用mysql存儲過程查詢數(shù)據(jù)實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2017-12-12
  • JdbcTemplate方法介紹與增刪改查操作實現(xiàn)

    JdbcTemplate方法介紹與增刪改查操作實現(xiàn)

    這篇文章主要給大家介紹了關于JdbcTemplate方法與增刪改查操作實現(xiàn)的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者使用JdbcTemplate具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
    2019-11-11
  • Java遞歸實現(xiàn)評論多級回復功能

    Java遞歸實現(xiàn)評論多級回復功能

    這篇文章主要介紹了Java遞歸實現(xiàn)評論多級回復功能,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-06-06

最新評論