详解Java中二叉树的基础概念(递归&迭代)

作者:爱干饭的猿 时间:2022-04-04 13:40:17 

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.2 二叉树的基本形态

详解Java中二叉树的基础概念(递归&迭代)

2.3 两种特殊的二叉树

详解Java中二叉树的基础概念(递归&迭代)

a.满二叉树:非子叶度都为2

b.完全二叉树:满二叉树缺了&ldquo;右下角&rdquo;

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 亦称先序遍历 )&mdash;&mdash; 访问根结点 ---> 根的左子树 ---> 根的右子树。

//先序遍历 : 根左右
   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)&mdash;&mdash; 根的左子树 ---> 根节点 ---> 根的右子树。

//中序遍历
   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)&mdash;&mdash; 根的左子树 ---> 根的右子树 ---> 根节点。

//后序遍历
   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);
               }
           }
       }
   }

3.二叉树完整代码

二叉树完整代码见下节:Java实现二叉树的示例代码(递归&迭代)

来源:https://blog.csdn.net/m0_62218217/article/details/123067319

标签:Java,二叉树,概念
0
投稿

猜你喜欢

  • Kotlin基础教程之伴生对象,getter,setter,内部,局部,匿名类,可变参数

    2021-08-05 03:56:02
  • java.lang.String类的使用

    2021-12-11 03:57:26
  • Spring Boot 搭建 ELK正确看日志的配置流程

    2022-08-28 17:13:46
  • Java处理InterruptedException异常的理论与实践

    2023-07-25 03:22:33
  • LeetCode程序员面试题之无重复字符的最长子串

    2021-09-12 05:26:36
  • Java 抽象类定义与方法实例详解

    2022-10-20 09:26:38
  • 入门JDK集合之HashMap解析

    2023-11-24 02:45:03
  • Android控件Tween动画(补间动画)实现方法示例

    2021-12-31 17:46:39
  • 半小时实现Java手撸网络爬虫框架(附完整源码)

    2022-11-23 15:59:35
  • C# BitArray(点矩阵)转换成int和string的方法实现

    2023-06-18 07:33:44
  • 简单实现Android本地音乐播放器

    2021-09-04 19:28:28
  • Java 迪杰斯特拉算法实现查找最短距离的实现

    2022-05-23 01:47:43
  • Java编程中实现归并排序算法的实例教程

    2023-09-10 07:40:41
  • 关于mybatis3中@SelectProvider的使用问题

    2021-11-20 17:03:19
  • c# 服务器上传木马监控代码(包含可疑文件)

    2022-07-08 10:02:45
  • 适用于WebForm Mvc的Pager分页组件C#实现

    2022-05-11 22:11:34
  • Android中实现地址栏输入网址能浏览该地址网页源码并操作访问网络

    2022-08-21 16:11:42
  • IntelliJ IDEA 好用插件之analyze inspect code详解

    2021-09-26 22:16:36
  • Spring中的两种代理JDK和CGLIB的区别浅谈

    2023-01-04 19:05:05
  • C#如何对多线程、多任务管理(demo)

    2023-05-19 06:00:21
  • asp之家 软件编程 m.aspxhome.com