结构与算法:二叉树与多叉树

开发 前端 算法
树形结构是一层次的嵌套结构。一个树形结构的外层和内层有相似的结构,所以这种结构多可以递归的表示。经典数据结构中的各种树状图是一种典型的树形结构:一颗树可以简单的表示为根, 左子树, 右子树。 左子树和右子树又有自己的子树。

一、树状结构

1、数组与链表

数组结构

数组存储是通过下标方式访问元素,查询速度快,如果数组元素是有序的,还可使用二分查找提高检索速度;如果添加新元素可能会导致多个下标移动,效率较低;

链表结构

链表存储元素,对于元素添加和删除效率高,但是遍历元素每次都需要从头结点开始,效率特别低;

树形结构能同时相对提高数据存储和读取的效率。

2、树结构概念

 

结构与算法:二叉树与多叉树
  • 根节点:树的根源,没有父节点的节点,如上图A节点;
  • 兄弟节点:拥有同一父节点的子节点。如图B与C点;
  • 叶子节点:没有子节点的节点。如图DEFG节点;
  • 树的高度:最大层数,如图为3层;
  • 路径:从root根节点找到指定节点的路线;

树形结构是一层次的嵌套结构。一个树形结构的外层和内层有相似的结构,所以这种结构多可以递归的表示。经典数据结构中的各种树状图是一种典型的树形结构:一颗树可以简单的表示为根, 左子树, 右子树。 左子树和右子树又有自己的子树。

二、二叉树模型

 

结构与算法:二叉树与多叉树

树的种类有很多,二叉树(BinaryTree)是树形结构的一个重要类型,每个节点最多只能有两个子节点的一种形式称为二叉树,二叉树的子节点分为左节点和右节点,许多实际问题抽象出来的数据结构往往是二叉树形式。

完全二叉树

 

结构与算法:二叉树与多叉树

二叉树的所有叶子节点都在最后一层或者倒数第二层,而且最后一层的叶子节点在左边连续,倒数第二 层的叶子节点在右边连续,我们称为完全二叉树

满二叉树

 

结构与算法:二叉树与多叉树

当二叉树的所有叶子节点都在最后一层,并且结点总数= 2^n -1 , n 为层数,则称为满二叉树。

平衡二叉树

 

结构与算法:二叉树与多叉树

平衡二叉树指的是,任意节点的子树的高度差的绝对值都小于等于1,并且左右两个子树都是一棵平衡二叉树,常见的符合平衡树的有,B树(多路平衡搜索树)、AVL树(二叉平衡搜索树)等。

二叉查找树

 

结构与算法:二叉树与多叉树

二叉查找树(BinarySearchTree)不但二叉树,同时满足一定的有序性:节点的左子节点比自己小,节点的右子节点比自己大。

三、二叉树编码

1、基础代码

节点代码

  1. class TreeNode { 
  2.     private String num ; 
  3.     private TreeNode leftNode ; 
  4.     private TreeNode rightNode ; 
  5.     public TreeNode(String num) { 
  6.         this.num = num; 
  7.     }    @Override 
  8.     public String toString() { 
  9.         return "TreeNode{num=" + num +'}'
  10.     }} 

树结构代码

  1. class BinaryTree01 { 
  2.     private TreeNode root ; 

2、遍历与查找

前序遍历查找

先处理当前结点的数据,再依次递归遍历左子树和右子树;

  1. public void prevTraverse() { 
  2.     // 输出父结点 
  3.     System.out.println(this); 
  4.     // 向左子树递归前序遍历 
  5.     if(this.leftNode != null) { 
  6.         this.leftNode.prevTraverse(); 
  7.     }    // 向右子树递归前序遍历 
  8.     if(this.rightNode != null) { 
  9.         this.rightNode.prevTraverse(); 
  10.     }}public TreeNode prevSearch(String num) {    //比较当前结点 
  11.     if(this.num.equals(num)) { 
  12.         return this ; 
  13.     }    // 递归遍历左子树查找 
  14.     TreeNode findNode = null
  15.     if(this.leftNode != null) { 
  16.         findNode = this.leftNode.prevSearch(num); 
  17.     }    // 左子树遍历命中 
  18.     if(findNode != null) { 
  19.         return findNode ; 
  20.     }    // 递归遍历右子树查找 
  21.     if(this.rightNode != null) { 
  22.         findNode = this.rightNode.prevSearch(num); 
  23.     }    return findNode ; 

中序遍历查找

先递归遍历左子树,再处理父节点,再递归遍历右子树

  1. public void midTraverse() { 
  2.     // 向左子树递归中序遍历 
  3.     if(this.leftNode != null) { 
  4.         this.leftNode.midTraverse(); 
  5.     }    // 输出父结点 
  6.     System.out.println(this); 
  7.     // 向右子树递归中序遍历 
  8.     if(this.rightNode != null) { 
  9.         this.rightNode.midTraverse(); 
  10.     }}public TreeNode midSearch(String num) {    // 递归遍历左子树查找 
  11.     TreeNode findNode = null
  12.     if(this.leftNode != null) { 
  13.         findNode = this.leftNode.midSearch(num); 
  14.     }    if(findNode != null) { 
  15.         return findNode ; 
  16.     }    // 比较当前结点 
  17.     if(this.num.equals(num)) { 
  18.         return this ; 
  19.     }    // 递归遍历右子树查找 
  20.     if(this.rightNode != null) { 
  21.         findNode = this.rightNode.midSearch(num); 
  22.     }    return findNode ; 

后序遍历查找

先递归遍历左子树,再递归遍历右子树,最后处理父节点;

  1. public void lastTraverse() { 
  2.     // 向左子树递归后序遍历 
  3.     if(this.leftNode != null) { 
  4.         this.leftNode.lastTraverse(); 
  5.     }    // 向右子树递归后序遍历 
  6.     if(this.rightNode != null) { 
  7.         this.rightNode.lastTraverse(); 
  8.     }    // 输出父结点 
  9.     System.out.println(this); 
  10. }public TreeNode lastSearch(String num) {    // 递归遍历左子树查找 
  11.     TreeNode findNode = null
  12.     if(this.leftNode != null) { 
  13.         findNode = this.leftNode.lastSearch(num); 
  14.     }    if(findNode != null) { 
  15.         return findNode ; 
  16.     }    // 递归遍历右子树查找 
  17.     if(this.rightNode != null) { 
  18.         findNode = this.rightNode.lastSearch(num); 
  19.     }    if(findNode != null) { 
  20.         return findNode ; 
  21.     }    // 比较当前结点 
  22.     if(this.num.equals(num)) { 
  23.         return this ; 
  24.     }    return null ; 

3、删除节点

如果当前删除的节点是叶子节点,则可以直接删除该节点;如果删除的节点是非叶子节点,则删除该节点树。

  1. public void deleteNode(String num) { 
  2.     // 判断左节点是否删除 
  3.     if(this.leftNode != null && this.leftNode.num.equals(num)) { 
  4.         this.leftNode = null ; 
  5.         return ; 
  6.     }    // 判断右节点是否删除 
  7.     if(this.rightNode != null && this.rightNode.num.equals(num)) { 
  8.         this.rightNode = null
  9.         return ; 
  10.     }    // 向左子树遍历进行递归删除 
  11.     if(this.leftNode != null) { 
  12.         this.leftNode.deleteNode(num); 
  13.     }    // 向右子树遍历进行递归删除 
  14.     if(this.rightNode != null) { 
  15.         this.rightNode.deleteNode(num); 
  16.     }} 

四、多叉树

 

结构与算法:二叉树与多叉树

多叉树是指一个父节点可以有多个子节点,但是一个子节点依旧遵循一个父节点定律,通常情况下,二叉树的实际应用高度太高,可以通过多叉树来简化对数据关系的描述。

例如:Linux文件系统,组织架构关系,角色菜单权限管理系统等,通常都基于多叉树来描述。

责任编辑:未丽燕 来源: 今日头条
相关推荐

2020-11-02 09:15:47

算法与数据结构

2020-04-27 07:05:58

二叉树左子树右子树

2013-07-15 16:35:55

二叉树迭代器

2021-09-29 10:19:00

算法平衡二叉树

2021-04-01 10:34:18

Java编程数据结构算法

2021-04-19 07:47:42

数据结构二叉树Tree

2021-04-20 08:37:14

数据结构二叉树

2021-03-19 10:25:12

Java数据结构算法

2021-04-28 20:12:27

数据结构创建

2021-03-22 09:00:22

Java数据结构算法

2018-03-15 08:31:57

二叉树存储结构

2021-09-15 07:56:32

二叉树层次遍历

2021-03-17 08:19:22

二叉树LeetCode

2022-12-26 00:51:33

双向链表二叉搜索树

2009-05-27 09:38:32

C#二叉树

2024-01-23 12:54:00

C++编程语言代码

2020-12-30 08:35:34

贪心算法监控

2021-09-28 06:28:51

二叉树公共祖先

2020-12-22 08:56:51

JavaScript数据结构前端

2009-08-11 13:29:57

C#二叉树遍历
点赞
收藏

51CTO技术栈公众号