搜尋
首頁Javajava教程Java中二元樹的基礎知識及概念是什麼?

    1. 樹型結構

    1.1概念

    樹是一種非線性的資料結構,它是由n ( n> ;=0 )個有限結點組成一個有層次關係的集合。把它叫做樹是因為它看 起來像一棵倒掛的樹,也就是說它是根朝上,而葉朝下的 。

    Java中二元樹的基礎知識及概念是什麼?

    1.2 概念(重要)

    a.節點的度數:此節點子樹的數目;如上圖:A的度數為6,J的度數為2

    b.樹的度:在該樹中,最大結點的度為該數的度數;如上圖:樹的度數為6

    c.葉節點(終端節點):度為0的節點(沒有子樹的節點)

    d.雙親結點/父節點:如上圖:D是H的父節點

    孩子節點/子節點:如上圖:H是D的子節點

    e.根節點:沒有雙親的節點;如上圖:A

    f.節點的層次:從根開始定義起,根為第1層,根的子節點為第2層,以此類推;

    g.樹的高度或深度:樹中節點的最大層次; 如上圖:樹的高度為4

    2. 二元樹(重點)

    2.1 概念

    每個節點最多只有兩顆子樹,度

    2.2 二元樹的基本形態

    Java中二元樹的基礎知識及概念是什麼?

    2.3 兩種特殊的二元樹

    Java中二元樹的基礎知識及概念是什麼?

    #a.滿二叉樹:非子葉度都為2

    #b.完全二元樹:滿二叉樹缺了「右下角」

    2.4 二元樹的性質

    a.滿一個二元樹

    1.高度為K,則有2^k-1個節點

    2.層次為K,則該層有2^(k-1)個節點

    3.邊數= 節點個數- 1

    4.度為0有n0個,度為2有n2個,則n0 = n2 1

    b.完全二元樹

    #1.有右孩子必有左孩子

    2.只可能有一個度為1的節點

    2.5 二元樹的儲存

    二元樹的儲存結構分為:順序存儲和類似鍊錶的鍊式存儲。

    順序儲存:只能存完全二元樹

    鍊式儲存:普通二元樹

    本次展示鍊式儲存

    二元樹的鍊式儲存是透過一個一個的節點引用起來的,常見的表示方式有二元和三叉表示方式,

    Java中二元樹的基礎知識及概念是什麼?

    以此圖為例, 具體如下:

    // 孩子表示法
    private static class TreeNode{
        char val;
        TreeNode left;
        TreeNode right;
     
        public TreeNode(char val) {
            this.val = val;
        }
    }

    初始化:

        public static TreeNode build(){
            TreeNode nodeA=new TreeNode('A');
            TreeNode nodeB=new TreeNode('B');
            TreeNode nodeC=new TreeNode('C');
            TreeNode nodeD=new TreeNode('D');
            TreeNode nodeE=new TreeNode('E');
            TreeNode nodeF=new TreeNode('F');
            TreeNode nodeG=new TreeNode('G');
            TreeNode nodeH=new TreeNode('H');
            nodeA.left=nodeB;
            nodeA.right=nodeC;
            nodeB.left=nodeD;
            nodeB.right=nodeE;
            nodeE.right=nodeH;
            nodeC.left=nodeF;
            nodeC.right=nodeG;
            return nodeA;
        }

    2.6 二元樹的基本操作

    2.6.1 二元樹的遍歷(遞歸)

    1. NLR :前序遍歷(Preorder Traversal 亦稱先序遍歷)—— 訪問根結點---> 根的左子樹---> 根的右子樹。

        //先序遍历 : 根左右
        public static void preOrder(TreeNode root){
            if(root==null){
                return;
            }
            System.out.print(root.val+" ");
            preOrder(root.left);
            preOrder(root.right);
        }

    2. LNR :中序遍歷 (Inorder Traversal)—— 根的左子樹 ---> 根節點 ---> 根的右子樹。

        //中序遍历
        public static void inOrder(TreeNode root){
            if(root==null){
                return;
            }
            preOrder(root.left);
            System.out.print(root.val+" ");
            preOrder(root.right);
        }

    3. LRN :後序遍歷 (Postorder Traversal)—— 根的左子樹 ---> 根的右子樹 ---> 根節點。

        //后序遍历
        public static void postOrder(TreeNode root){
            if(root==null){
                return;
            }
            preOrder(root.left);
            preOrder(root.right);
            System.out.print(root.val+" ");
        }

    2.6.2 二元樹的遍歷(迭代)

    1.前序遍歷

        //方法2(迭代)
        //先序遍历 (迭代)
        public static void preOrderNonRecursion(TreeNode root){
            if(root==null){
                return ;
            }
            Deque<TreeNode> stack=new LinkedList<>();
            stack.push(root);
            while (!stack.isEmpty()){
                TreeNode cur=stack.pop();
                System.out.print(cur.val+" ");
                if(cur.right!=null){
                    stack.push(cur.right);
                }
                if(cur.left!=null){
                    stack.push(cur.left);
                }
            }
        }

    2.中序遍歷

        //方法2(迭代)
        //中序遍历 (迭代)
        public static void inorderTraversalNonRecursion(TreeNode root) {
            if(root==null){
                return ;
            }
     
            Deque<TreeNode> stack=new LinkedList<>();
            // 当前走到的节点
            TreeNode cur=root;
            while (!stack.isEmpty() || cur!=null){
                // 不管三七二十一,先一路向左走到根儿~
                while (cur!=null){
                    stack.push(cur);
                    cur=cur.left;
                }
                // 此时cur为空,说明走到了null,此时栈顶就存放了左树为空的节点
                cur=stack.pop();
                System.out.print(cur.val+" ");
                // 继续访问右子树
                cur=cur.right;
            }
        }

    3.後序遍歷

        //方法2(迭代)
        //后序遍历 (迭代)
        public static void postOrderNonRecursion(TreeNode root){
            if(root==null){
                return;
            }
            Deque<TreeNode> stack=new LinkedList<>();
            TreeNode cur=root;
            TreeNode prev=null;
     
            while (!stack.isEmpty() || cur!=null){
                while (cur!=null){
                    stack.push(cur);
                    cur=cur.left;
                }
     
                cur=stack.pop();
                if(cur.right==null || prev==cur.right){
                    System.out.print(cur.val+" ");
                    prev=cur;
                    cur=null;
                }else {
                    stack.push(cur);
                    cur=cur.right;
                }
            }
        }

    2.6.3 二元樹的基本操作

    1.求結點個數(遞歸&迭代)

        //方法1(递归)
        //传入一颗二叉树的根节点,就能统计出当前二叉树中一共有多少个节点,返回节点数
        //此时的访问就不再是输出节点值,而是计数器 + 1操作
        public static int getNodes(TreeNode root){
            if(root==null){
                return 0;
            }
            return 1+getNodes(root.left)+getNodes(root.right);
        }
     
        //方法2(迭代)
        //使用层序遍历来统计当前树中的节点个数
        public static int getNodesNoRecursion(TreeNode root){
            if(root==null){
                return 0;
            }
            int size=0;
            Deque<TreeNode> queue=new LinkedList<>();
            queue.offer(root);
            while (!queue.isEmpty()) {
                TreeNode cur = queue.poll();
                size++;
                if (cur.left != null) {
                    queue.offer(cur.left);
                }
                if (cur.right != null) {
                    queue.offer(cur.right);
                }
            }
            return size;
        }

    2.求葉子結點個數(遞歸&迭代)

        //方法1(递归)
        //传入一颗二叉树的根节点,就能统计出当前二叉树的叶子结点个数
        public static int getLeafNodes(TreeNode root){
            if(root==null){
                return 0;
            }
            if(root.left==null && root.right==null){
                return 1;
            }
            return getLeafNodes(root.left)+getLeafNodes(root.right);
        }
     
        //方法2(迭代)
        //使用层序遍历来统计叶子结点的个数
        public static int getLeafNodesNoRecursion(TreeNode root){
            if(root==null){
                return 0;
            }
            int size=0;
            Deque<TreeNode> queue=new LinkedList<>();
            queue.offer(root);
            while (!queue.isEmpty()){
                TreeNode cur=queue.poll();
                if(cur.left==null && cur.right==null){
                    size++;
                }
                if(cur.left!=null){
                    queue.offer(cur.left);
                }
                if(cur.right!=null){
                    queue.offer(cur.right);
                }
            }
            return size;
        }

    3.求第k 層結點個數

        //求出以root为根节点的二叉树第k层的节点个数
        public static int getKLevelNodes(TreeNode root,int k){
            if(root==null || k<=0){
                return 0;
            }
            if(k==1){
                return 1;
            }
            return getKLevelNodes(root.left,k-1)+getKLevelNodes(root.right,k-1);
        }

    4.求樹的高度

        //传入一个以root为根节点的二叉树,就能求出该树的高度
        public static int height(TreeNode root){
            if(root==null){
                return 0;
            }
            return 1+ Math.max(height(root.left),height(root.right));
        }

    5.判斷二元樹數中是否存在值為value的節點

        //判断当前以root为根节点的二叉树中是否包含指定元素val,
        //若存在返回true,不存在返回false
        public static boolean contains(TreeNode root,char value){
            if(root==null){
                return false;
            }
            if(root.val==value){
                return true;
            }
            return contains(root.left,value) || contains(root.right,value);
        }

    2.7 二元樹的層序遍歷

        //层序遍历
        public static void levelOrder(TreeNode root) {
            if(root==null){
                return ;
            }
     
            // 借助队列来实现遍历过程
            Deque<TreeNode> queue =new LinkedList<>();
            queue.offer(root);
            while (!queue.isEmpty()){
                int size=queue.size();
                for (int i = 0; i < size; i++) {
                    TreeNode cur=queue.poll();
                    System.out.print(cur.val+" ");
                    if(cur.left!=null){
                        queue.offer(cur.left);
                    }
                    if(cur.right!=null){
                        queue.offer(cur.right);
                    }
                }
            }
        }

    以上是Java中二元樹的基礎知識及概念是什麼?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

    陳述
    本文轉載於:亿速云。如有侵權,請聯絡admin@php.cn刪除
    在Java中如何在項目啟動時動態修改easypoi中@Excel註解的savePath參數?在Java中如何在項目啟動時動態修改easypoi中@Excel註解的savePath參數?Apr 19, 2025 pm 02:09 PM

    在Java中如何動態配置實體類註解的參數在開發過程中,我們經常會遇到需要根據不同環境動態配置註解參數的�...

    在YARN上提交PyFlink作業時,為什麼會報錯無法找到Python腳本?在YARN上提交PyFlink作業時,為什麼會報錯無法找到Python腳本?Apr 19, 2025 pm 02:06 PM

    在YARN上提交PyFlink作業時報錯無法找到Python腳本的原因分析當你嘗試通過YARN提交一個PyFlink作業時,可能會遇到�...

    Spring Boot項目中調用第三方接口,字段名大小寫和getter方法不一致導致數據傳輸失敗怎麼辦?Spring Boot項目中調用第三方接口,字段名大小寫和getter方法不一致導致數據傳輸失敗怎麼辦?Apr 19, 2025 pm 02:03 PM

    在SpringBoot項目中調用第三方接口傳輸數據時遇到的難題本文將針對一個Spring...

    如何將名字轉換為數字以實現群組內排序?如何將名字轉換為數字以實現群組內排序?Apr 19, 2025 pm 01:57 PM

    如何將名字轉為數字以實現群組內排序?在群組中排序用戶時,常常需要將用戶的名字轉化為數字,以便在不同...

    在Java遠程調試中,如何正確獲取遠程服務器上的常量值?在Java遠程調試中,如何正確獲取遠程服務器上的常量值?Apr 19, 2025 pm 01:54 PM

    Java遠程調試中常量獲取的疑問解答在使用Java進行遠程調試時,許多開發者可能會遇到一些難以理解的現象。其�...

    在後端開發中,如何區分service層和dao層的職責?在後端開發中,如何區分service層和dao層的職責?Apr 19, 2025 pm 01:51 PM

    探討後端開發中的分層架構在後端開發中,分層架構是一種常見的設計模式,通常包括controller、service和dao三層�...

    See all articles

    熱AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智慧驅動的應用程序,用於創建逼真的裸體照片

    AI Clothes Remover

    AI Clothes Remover

    用於從照片中去除衣服的線上人工智慧工具。

    Undress AI Tool

    Undress AI Tool

    免費脫衣圖片

    Clothoff.io

    Clothoff.io

    AI脫衣器

    AI Hentai Generator

    AI Hentai Generator

    免費產生 AI 無盡。

    熱工具

    SecLists

    SecLists

    SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

    WebStorm Mac版

    WebStorm Mac版

    好用的JavaScript開發工具

    ZendStudio 13.5.1 Mac

    ZendStudio 13.5.1 Mac

    強大的PHP整合開發環境

    Safe Exam Browser

    Safe Exam Browser

    Safe Exam Browser是一個安全的瀏覽器環境,安全地進行線上考試。該軟體將任何電腦變成一個安全的工作站。它控制對任何實用工具的訪問,並防止學生使用未經授權的資源。

    MinGW - Minimalist GNU for Windows

    MinGW - Minimalist GNU for Windows

    這個專案正在遷移到osdn.net/projects/mingw的過程中,你可以繼續在那裡關注我們。 MinGW:GNU編譯器集合(GCC)的本機Windows移植版本,可自由分發的導入函式庫和用於建置本機Windows應用程式的頭檔;包括對MSVC執行時間的擴展,以支援C99功能。 MinGW的所有軟體都可以在64位元Windows平台上運作。