如何用 JS 实现二叉堆

开发 前端
二叉树(Binary Tree)是一种树形结构,它的特点是每个节点最多只有两个分支节点,一棵二叉树通常由根节点、分支节点、叶子节点组成,如下图所示。每个分支节点也常常被称作为一棵子树,而二叉堆是一种特殊的树,它属于完全二叉树。

[[384564]]

前言

二叉树(Binary Tree)是一种树形结构,它的特点是每个节点最多只有两个分支节点,一棵二叉树通常由根节点、分支节点、叶子节点组成,如下图所示。每个分支节点也常常被称作为一棵子树,而二叉堆是一种特殊的树,它属于完全二叉树。

二叉树与二叉堆的关系

在日常工作中会遇到很多数组的操作,比如排序等。那么理解二叉堆的实现对以后的开发效率会有所提升,下面就简单介绍一下什么是二叉树,什么是二叉堆。

二叉树特征

  • 根节点:二叉树最顶层的节点
  • 分支节点:除了根节点以外且拥有叶子节点
  • 叶子节点:除了自身,没有其他子节点

在二叉树中,我们常常还会用父节点和子节点来描述,比如上图中左侧节点 2 为 6 和 3 的父节点,反之 6 和 3 是 2 子节点。

二叉树分类

二叉树分为满二叉树(full binary tree)和完全二叉树(complete binary tree)。

  • 满二叉树:一棵深度为 k 且有 2 ^ k - 1个节点的二叉树称为满二叉树
  • 完全二叉树:完全二叉树是指最后一层左边是满的,右边可能满也可能不满,然后其余层都是满的二叉树称为完全二叉树(满二叉树也是一种完全二叉树)

二叉树结构

从图中我们可以看出二叉树是从上到下依次排列下来,可想而知可以用一个数组来表示二叉树的结构,从下标 index( 0 - 8 ) 从上到下依次排列。

 

  • 二叉树左侧节点表达式 index * 2 + 1。例如:以根节点为例求左侧节点,根节点的下标为0,则左侧节点的序数是1 ,对应数组中的值为1
  • 二叉树右侧节点表达式 index * 2 + 2。例如:以根节点为例求右侧节点,根节点的下标为0,则右侧节点的序数是2 ,对应数组中的值为 8
  • 二叉树叶子节点表达式 序数 >= floor( N / 2 )都是叶子节点(N是数组的长度)。例如:floor( 9 / 2 ) = 4 ,则从下标 4 开始的值都为叶子节点

二叉堆特征

二叉堆是一个完全二叉树,父节点与子节点要保持固定的序关系,并且每个节点的左子树和右子树都是一个二叉堆。

从上图可以看出

  • 图一:每个父节点大于子节点或等于子节点,满足二叉堆的性质
  • 图二:其中有一个父节点小于子节点则不满足二叉堆性质

二叉堆分类

​ 二叉堆根据排序不同,可以分为最大堆和最小堆

  • 最大堆:根节点的键值是所有堆节点键值中最大者,且每个父节点的值都比子节点的值大
  • 最小堆:根节点的键值是所有堆节点键值中最小者,且每个父节点的值都比子节点的值小

如何实现二叉堆

通过上面的讲述想必大家对二叉堆有了一定的理解,那么接下来就是如何实现。以最大堆为例,首先要初始化数组然后通过交换位置形成最大堆。

初始化二叉堆

从上面描述,我们可以知道二叉堆其实就是一个数组,那么初始化就非常简单了。

 

  1. class Heap{ 
  2.   constructor(arr){ 
  3.     this.data = [...arr]; 
  4.     this.size = this.data.length; 
  5.   } 

父子节点交换位置

图一中 2 作为父节点小于子节点,很显然不符合最大堆性质。maxHeapify 函数可以把每个不符合最大堆性质的节点调换位置,从而满足最大堆性质的数组。

调整步骤:

  • 调整分支节点 2 的位置(不满足最大堆性质)
  • 获取父节点 2 的左右节点 ( 12 , 5 ) ,从 ( 2 , 15 , 5 ) 中进行比较
  • 找出最大的节点与父节点进行交换,如果该节点本身为最大节点则停止操作
  • 重复 step2 的操作,从 2 , 4 , 7 中找出最大值与 2 做交换(递归)

 

  1. maxHeapify(i) { 
  2.   let max = i; 
  3.  
  4.   if(i >= this.size){ 
  5.     return
  6.   } 
  7.   // 当前序号的左节点 
  8.   const l = i * 2 + 1; 
  9.   // 当前需要的右节点 
  10.   const r = i * 2 + 2; 
  11.  
  12.   // 求当前节点与其左右节点三者中的最大值 
  13.   if(l < this.size && this.data[l] > this.data[max]){ 
  14.     max = l; 
  15.   } 
  16.   if(r < this.size && this.data[r] > this.data[max]){ 
  17.     max = r; 
  18.   } 
  19.  
  20.   // 最终max节点是其本身,则已经满足最大堆性质,停止操作 
  21.   if(max === i) { 
  22.     return
  23.   } 
  24.  
  25.   // 父节点与最大值节点做交换 
  26.   const t = this.data[i]; 
  27.   this.data[i] = this.data[max]; 
  28.   this.data[max] = t; 
  29.  
  30.   // 递归向下继续执行 
  31.   return this.maxHeapify(max); 

形成最大堆

我们可以看到,初始化是由一个数组组成,以下图为例很显然并不会满足最大堆的性质,上述 maxHeapify 函数只是对某一个节点作出对调,无法对整个数组进行重构,所以我们要依次对数组进行递归重构。

  • 找到所有分支节点 Math.floor( N / 2 )(不包括叶子节点)
  • 将找到的子节点进行 maxHeapify 操作

 

  1. rebuildHeap(){ 
  2.   // 叶子节点 
  3.   const L = Math.floor(this.size / 2); 
  4.   for(let i = L - 1; i >= 0; i--){ 
  5.     this.maxHeapify(i); 
  6.   } 

生成一个升序的数组

  • swap 函数交换首位位置
  • 将最后一个从堆中拿出相当于 size - 1
  • 执行 maxHeapify 函数进行根节点比较找出最大值进行交换
  • 最终 data 会变成一个升序的数组

 

  1. sort() { 
  2.   for(let i = this.size - 1; i > 0; i--){ 
  3.     swap(this.data, 0, i); 
  4.     this.size--; 
  5.     this.maxHeapify(0); 
  6.   } 

插入方法

Insert 函数作为插入节点函数,首先

  1. 往 data 结尾插入节点
  2. 因为节点追加,size + 1
  3. 因为一个父节点拥有 2 个子节点,我们可以根据这个性质通过 isHeap 函数获取第一个叶子节点,可以通过第一个叶子节点获取新插入的节点,然后进行 3 个值的对比,找出最大值,判断插入的节点。如果跟父节点相同则不进行重构(相等满足二叉堆性质),否则进行 rebuildHeap 重构堆

 

  1. isHeap() { 
  2.   const L = Math.floor(this.size / 2); 
  3.   for (let i = L - 1; i >= 0; i--) { 
  4.     const l = this.data[left(i)] || Number.MIN_SAFE_INTEGER; 
  5.     const r = this.data[right(i)] || Number.MIN_SAFE_INTEGER; 
  6.  
  7.     const max = Math.max(this.data[i], l, r); 
  8.  
  9.     if (max !== this.data[i]) { 
  10.       return false
  11.     } 
  12.     return true
  13.   } 
  14. insert(key) { 
  15.   this.data[this.size] = key
  16.   this.size++ 
  17.   if (this.isHeap()) { 
  18.     return
  19.   } 
  20.   this.rebuildHeap(); 

删除方法

delete 函数作为删除节点,首先

  • 删除传入index的节点
  • 因为节点删除,size - 1
  • 重复上面插入节点的操作

 

  1. delete(index) { 
  2.   if (index >= this.size) { 
  3.     return
  4.   } 
  5.   this.data.splice(index, 1); 
  6.   this.size--; 
  7.   if (this.isHeap()) { 
  8.     return
  9.   } 
  10.   this.rebuildHeap(); 

完整代码

 

  1. /** 
  2.  * 最大堆 
  3.  */ 
  4.  
  5. function left(i) { 
  6.   return (i * 2) + 1; 
  7.  
  8. function right(i) { 
  9.   return (i * 2) + 2; 
  10.  
  11. function swap(A, i, j) { 
  12.   const t = A[i]; 
  13.   A[i] = A[j]; 
  14.   A[j] = t; 
  15.  
  16. class Heap { 
  17.   constructor(arr) { 
  18.     this.data = [...arr]; 
  19.     this.size = this.data.length; 
  20.     this.rebuildHeap = this.rebuildHeap.bind(this); 
  21.     this.isHeap = this.isHeap.bind(this); 
  22.     this.sort = this.sort.bind(this); 
  23.     this.insert = this.insert.bind(this); 
  24.     this.delete = this.delete.bind(this); 
  25.     this.maxHeapify = this.maxHeapify.bind(this); 
  26.   } 
  27.  
  28.   /** 
  29.    * 重构堆,形成最大堆 
  30.    */ 
  31.   rebuildHeap() { 
  32.     const L = Math.floor(this.size / 2); 
  33.     for (let i = L - 1; i >= 0; i--) { 
  34.       this.maxHeapify(i); 
  35.     } 
  36.   } 
  37.  
  38.   isHeap() { 
  39.     const L = Math.floor(this.size / 2); 
  40.     for (let i = L - 1; i >= 0; i--) { 
  41.       const l = this.data[left(i)] || Number.MIN_SAFE_INTEGER; 
  42.       const r = this.data[right(i)] || Number.MIN_SAFE_INTEGER; 
  43.  
  44.       const max = Math.max(this.data[i], l, r); 
  45.  
  46.       if (max !== this.data[i]) { 
  47.         return false
  48.       } 
  49.       return true
  50.     } 
  51.   } 
  52.  
  53.   sort() { 
  54.     for (let i = this.size - 1; i > 0; i--) { 
  55.       swap(this.data, 0, i); 
  56.       this.size--; 
  57.       this.maxHeapify(0); 
  58.     } 
  59.   } 
  60.  
  61.   insert(key) { 
  62.     this.data[this.size++] = key
  63.     if (this.isHeap()) { 
  64.       return
  65.     } 
  66.     this.rebuildHeap(); 
  67.   } 
  68.  
  69.   delete(index) { 
  70.     if (index >= this.size) { 
  71.       return
  72.     } 
  73.     this.data.splice(index, 1); 
  74.     this.size--; 
  75.     if (this.isHeap()) { 
  76.       return
  77.     } 
  78.     this.rebuildHeap(); 
  79.   } 
  80.  
  81.   /** 
  82.    * 交换父子节点位置,符合最大堆特征 
  83.    * @param {*} i 
  84.    */ 
  85.   maxHeapify(i) { 
  86.     let max = i; 
  87.  
  88.     if (i >= this.size) { 
  89.       return
  90.     } 
  91.  
  92.     // 求左右节点中较大的序号 
  93.     const l = left(i); 
  94.     const r = right(i); 
  95.     if (l < this.size && this.data[l] > this.data[max]) { 
  96.       max = l; 
  97.     } 
  98.  
  99.     if (r < this.size && this.data[r] > this.data[max]) { 
  100.       max = r; 
  101.     } 
  102.  
  103.     // 如果当前节点最大,已经是最大堆 
  104.     if (max === i) { 
  105.       return
  106.     } 
  107.  
  108.     swap(this.data, i, max); 
  109.  
  110.     // 递归向下继续执行 
  111.     return this.maxHeapify(max); 
  112.   } 
  113.  
  114. module.exports = Heap; 

示例

相信通过上面的讲述大家对最大堆的实现已经有了一定的理解,我们可以利用这个来进行排序。

 

  1. const arr = [15, 12, 8, 2, 5, 2, 3, 4, 7]; 
  2. const fun = new Heap(arr); 
  3. fun.rebuildHeap(); // 形成最大堆的结构 
  4. fun.sort();// 通过排序,生成一个升序的数组 
  5. console.log(fun.data) // [2, 2, 3, 4, 5, 7, 8, 12, 15] 

总结

文章中主要讲述了二叉树、二叉堆的概念,然后通过代码实现二叉堆。我们可以通过二叉堆来做排序和优先级队列等。

责任编辑:未丽燕 来源: ZooTeam Blog
相关推荐

2020-11-23 08:53:34

堆Heap

2020-08-31 07:43:58

二叉堆大顶堆存储

2021-05-06 05:29:32

二叉堆数据结构算法

2023-04-06 07:39:48

2020-10-11 16:56:48

二叉搜索树代码开发

2020-04-27 07:05:58

二叉树左子树右子树

2021-04-20 08:37:14

数据结构二叉树

2023-07-31 08:01:13

二叉搜索测试

2021-09-03 08:58:00

二叉搜索树节点

2020-12-22 08:56:51

JavaScript数据结构前端

2009-08-11 13:29:57

C#二叉树遍历

2020-12-11 09:49:29

二叉树搜索树数据

2022-10-12 23:25:17

二叉树父节点根节点

2009-05-27 09:38:32

C#二叉树

2024-01-23 12:54:00

C++编程语言代码

2022-12-26 00:51:33

双向链表二叉搜索树

2021-09-29 10:19:00

算法平衡二叉树

2021-04-19 07:47:42

数据结构二叉树Tree

2021-05-09 20:22:41

顺序查找二叉查找数据结构

2021-03-17 08:19:22

二叉树LeetCode
点赞
收藏

51CTO技术栈公众号