堆排序(向下调整法,向上调整法详解)

news/2024/11/29 6:28:54/

目录

一、 二叉树的顺序结构

二、 堆的概念及结构

三、数组存储、顺序存储的规律

此处可能会有疑问,左右孩子的父节点计算为什么可以归纳为一个结论了?

四、大小堆解释

五、大小堆的实现(向上和向下调整法)

5.11向上调整法

 ​编辑

5.12向上调整法时间复杂度计算

5.21向下调整法

5.22向下调整法的时间复杂度计算

​编辑

六、堆排序的实现

代码如下:


一、 二叉树的顺序结构

普通的二叉树是不适合用数组来存储的,因为可能会存在大量的空间浪费。而完全二叉树更适合使用顺序结构存储。现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储,需要注意的是这里的堆和操作系统虚拟进程地址空间中的堆是两回事,一个是数据结构,一个是操作系统中管理内存的一块区域分段。

二、 堆的概念及结构

如果有一个关键码的集合k ={ k_0{}^{},k_1{}^{},k_2{}^{},...,k_n{}^{} },把它的所有元素按完全二叉树的顺序存储方式存储在一个一维数组中,并满足:

K_i <= K_{2*i+1}K_i <= K_{2*i+2}(K_i >= K_{2*i+2}K_i>=K_{2*i+2})i = 0,
2…,则称为小堆(或大堆)。将根节点最大的堆叫做最大堆或大根堆,根节点最小的堆叫做最小堆或小根堆。

  • 堆中某个节点的值总是不大于或不小于其父节点的值;
  • 堆总是一棵完全二叉树(完全二叉树是从满二叉树中最后一层连续删除若干结点(只能从最右侧删除)后得到的二叉树。)。

要满足K_i <= K_{2*i+1}K_i <= K_{2*i+2}(K_i >= K_{2*i+2}K_i>=K_{2*i+2})的原因。

三、数组存储、顺序存储的规律

如果要用数组存储二叉树,那么必须要符合顺序存储中父子存储的规律

此处可能会有疑问,左右孩子的父节点计算为什么可以归纳为一个结论了?

  • 一个左子节点索引 leftchild 和一个右子节点索引 rightchild,并且它们共享同一个父节点时,这意味着 rightchild = leftchild + 1。现在,如果你用上述公式来计算它们的父节点索引:
  • 对于左子节点:parent = (leftchild - 1) / 2
  • 对于右子节点:parent = (rightchild - 2) / 2但因为 rightchild = leftchild + 1,所以:
  • parent = ((leftchild + 1) - 2) / 2
  • parent = (leftchild - 1) / 2
  • 并且由于(int)3/(int)2 = (int)1,这一向下取整的性质,所以在这一计算过程中不会出现浮点数的情况
  • 你可以看到,无论你是从左子节点还是右子节点开始计算,你都得到了相同的父节点索引。

但是数组存储二叉树是有要求的。如果不符合该规律,那么得设置空节点去代替缺失的节点(因为要满足下标的规律才能方便查找),那么使用太多的空节点会造成空间的浪费。

结论:数组存储只适合完全二叉树和满二叉树

四、大小堆解释

 

堆并非是一定有序的 :左孩子与右孩子之间没有大小关系

  • 大堆:在最大堆中,父节点的值总是大于或等于其子节点的值。但是,左孩子和右孩子之间并没有固定的大小关系。也就是说,左孩子可以大于、小于或等于右孩子,这都不会违反最大堆的定义。
  • 也就是说,对于给定的节点i,其值应满足:array[i] >= array[2i + 1] 且 array[i] >= array[2i + 2]。
  • 小堆:在最小堆中,父节点的值总是小于或等于其子节点的值。同样地,左孩子和右孩子之间的大小关系是不确定的。
  • 也就是说,对于给定的节点i,其值应满足:array[i] <= array[2i + 1] 且 array[i] <= array[2i + 2]。
  • 这里的“2i + 1”和“2i + 2”分别表示节点i的左子节点和右子节点在数组中的位置(假设数组是从0开始索引的)。

这种特性使得堆成为一种非常有效的数据结构,特别是在实现优先队列等应用中。堆可以在对数时间内完成插入和删除最大(或最小)元素的操作,这是因为它不需要保持整个结构的完全排序。

举个例子:

    10  /   \  5     8  / \   / \  
2   3 6   7

在这个堆中,父节点的值总是大于或等于其子节点的值。但是,你可以看到左孩子和右孩子之间的大小关系是不一致的。例如,5的左孩子是2,右孩子是3,而8的左孩子是6,右孩子是7。这里并没有规定左孩子必须大于或小于右孩子。 

五、大小堆的实现(向上和向下调整法)

void Swap(HPDataType* px,HPDataType* py)
{*py ^= *px;*px ^= *py;*py ^= *px;
}

5.11向上调整法

目的:
当向堆中插入新元素时,为了维护堆的性质,需要对该元素进行向上调整。向上调整法就是从新插入的节点开始,通过与其父节点的比较和交换,确保该节点的值不大于(对于大根堆)或不小于(对于小根堆)其父节点的值。

步骤:

  1. 插入数据
  2. 与自己的父亲比较
  3. 交换/不交换
  4. 交换:孩子来到父亲位置,父亲来到自己父亲的位置。

判断条件:a[child] > a[parent]

结束循环条件:child > 0  (确保不是根节点)

时间复杂度:O(logN),其中N是堆中元素的数量。因为每次调整都涉及沿着树的一条路径向上移动,而树的深度为logN。

 

 void AdjustUP(HPDataType* a, int n, int parent)参数的意义:

  • HPDataType是一个自定义的数据类型,代表堆中存储的数据的类型int,a是一个指向HPDataType类型数组的指针,这个数组存储了堆中的所有元素。
  • child表示当前要进行向上调整的节点的索引。在堆排序中,当我们向堆中插入一个新的元素时,这个新元素通常被放置在数组的末尾,然后可能需要通过向上调整来确保它满足堆的性质。child就是这个新插入元素的索引。
void AdjustUp(HPDataType* a, int child)
{int parent = (child - 1) / 2;// 获取父节点索引//while (parent >= 0)while(child > 0)// 确保不是根节点{ //if (a[child] < a[parent])// 孩子小于于父亲,需要交换,向下调整法if (a[child] > a[parent])// 孩子大于父亲,需要交换, 向上调整法// 如果孩子节点大于父节点,则交换{Swap(&a[child], &a[parent]);child = parent;// 移动到父节点parent = (parent - 1) / 2;}else {break;}}}

5.12向上调整法时间复杂度计算

可得高度与向上调整的关系 F(h)=2^h*(h-2)+2

时间复杂度F(N)=(N+1)*(log(N+1)-2)+2

5.21向下调整法

目的:
当从堆中移除元素(通常是堆顶元素)后,为了维护堆的性质,需要对剩余的元素进行重新调整。向下调整法就是从父节点开始,通过与其子节点的比较和交换,确保父节点的值不大于(对于大根堆)或不小于(对于小根堆)其子节点的值。

步骤:

  1. 删除堆顶元素
  2. 堆顶元素与最后一个元素交换
  3. 删除最后一个元素
  4. 堆顶元素与左右两个孩子(最小/最大的孩子比较)
  5. 判断交换/不交换
  6. 交换:父亲来到孩子位置,孩子来到自己孩子的位置

判断条件:child + 1 < n && a[child + 1] < a[child]

结束循环条件:child < n(确保左孩子存在)

时间复杂度:O(logN),其中N是堆中元素的数量。因为每次调整都涉及沿着树的一条路径向下移动,而树的深度为logN。

如何删除堆顶数据后插入数据?

向下调整法

如果直接挪动覆盖:操作的时间复杂度太大,关系太乱,不如重新建堆

向下调整法:

 void AdjustDown(HPDataType* a, int n, int parent)参数的意义:

  • HPDataType是一个自定义的数据类型,代表堆中存储的数据的类型int,a是一个指向HPDataType类型数组的指针,这个数组存储了堆中的所有元素。
  • n表示堆中当前最后一个元素的下标。在堆排序的过程中,堆的大小可能会变化,因为我们会不断地从堆中移除元素。这个参数确保我们知道何时停止向下调整,即当child索引超过最后一个下标时。
  • parent表示当前要调整的节点的索引。在堆排序中,当我们从堆中移除堆顶元素并与堆的最后一个元素交换时,我们需要对新的堆顶元素进行向下调整以确保堆的性质得到维护。parent就是这个需要进行调整的节点的索引。
// 向下调整算法(用于删除或构建堆时维护堆)  
void AdjustDown(HPDataType* a, int n, int parent)
{int child = parent * 2 + 1; // 获取左孩子索引  while (child < n) // 确保左孩子存在  {// 如果右孩子存在且大于左孩子,则选择右孩子  if (child + 1 < n && a[child + 1] > a[child]){++child; // 选择右孩子  }// 如果孩子节点大于父节点,则交换  if (a[child] > a[parent]){Swap(&a[child], &a[parent]);parent = child; // 移动到孩子节点  child = parent * 2 + 1; // 获取新的左孩子索引  }else {break; // 不需要交换,退出循环  }}
}

5.22向下调整法的时间复杂度计算

可得高度与向下调整次数的关系 F(h)=2^{h}-h-1

可得时间复杂度:F(N) = N-log(N+1)

六、堆排序的实现

有一个数列,请用堆排序升序排列

如果使用向下调整法建小堆,先把0视为堆根,0和3交换,然后当3视为堆根时:

所以要建大堆:

堆排序的时间复杂度与向上调整法建堆时差不多

子节点大于父节点时交换,建大堆,升序,保证父节点小于子节点

子节点小于父节点时交换,建小堆,降序,保证父节点大于子节点 

代码如下:

#include<bits/stdc++.h>
using namespace std;void Swap(int* px, int* py)
{*py ^= *px;*px ^= *py;*py ^= *px;
}
  • 该函数是堆排序的核心,用于调整堆的结构,确保其满足堆的性质(父节点小于其子节点,这是小根堆;反之则是大根堆。这里的代码是小根堆的实现)。
  • 接收三个参数:一个整数数组a、数组的长度n以及要调整的父节点的索引parent。
  • 首先,计算左孩子的索引child。
  • 然后,通过循环,比较父节点和孩子节点的大小。如果存在右孩子且右孩子的值小于左孩子,则选择右孩子作为更小的孩子。
  • 如果更小的孩子的值小于父节点,则交换它们的值,并将parent移动到新的位置,再次检查新的子节点。
  • 如果子节点不小于父节点,则循环终止,调整完成。
// 向下调整算法(用于删除或构建堆时维护堆)  
void AdjustDown(int* a, int n, int parent)
{int child = parent * 2 + 1; // 获取左孩子索引  while (child < n) // 确保左孩子存在  {// 如果右孩子存在且大于左孩子,则选择右孩子  if (child + 1 < n && a[child + 1] < a[child]){++child; // 选择右孩子  }// 如果孩子节点大于父节点,则交换  if (a[child] < a[parent]){Swap(&a[child], &a[parent]);parent = child; // 移动到孩子节点  child = parent * 2 + 1; // 获取新的左孩子索引  }else {break; // 不需要交换,退出循环  }}
}
  • 首先,对数组a建立一个小根堆。从最后一个非叶子节点开始(即索引为(n-1-1)/2的节点),调用AdjustDown函数调整每个子树。
  • 一旦堆建立完毕,进入循环:将堆顶元素(数组的第一个元素)与堆的最后一个元素交换,然后重新调整剩下的元素为堆,但每次调整的范围都减小一个(即排除掉最后一个元素)。
  • 循环继续,直到堆的大小为1,此时数组已经完全排序。
void HeapSort(int* a, int n)
{// a数组直接建堆 O(N)for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, n, i);}// O(N*logN)int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);// 首尾交换AdjustDown(a, end, 0);// 向下调整--end;}
}

这个函数首先通过AdjustDown函数将数组转化为最大堆。然后,它反复地将堆的根节点(即最大元素)与堆的最后一个节点交换,并重新调整堆,直到整个数组被排序。

int main()
{int a[] = { 3,6,1,5,8,9,2,7,4,0 };HeapSort(a, sizeof(a) / sizeof(int));for (int i = 0; i < 10; i++)printf("%d", a[i]);return 0;
}

今天就先到这了!!!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。


http://www.ppmy.cn/news/1387359.html

相关文章

2.1HTML5基本结构

HTML5实际上不算是一种编程语言&#xff0c;而是一种标记语言。HTML5文件是由一系列成对出现的元素标签嵌套组合而成&#xff0c;这些标签以<元素名>的形式出现&#xff0c;用于标记文本内容的含义。浏览器通过元素标签解析文本内容并将结果显示在网页上&#xff0c;而元…

加拿大光量子计算公司Xanadu入局英国多企业量子合作计划

内容来源&#xff1a;量子前哨&#xff08;ID&#xff1a;Qforepost&#xff09; 编辑丨慕一 编译/排版丨沛贤 深度好文&#xff1a;1200字丨8分钟阅读 英国航空发动机制造商罗尔斯罗伊斯&#xff08;Rolls-Royce&#xff09;、英国量子计算公司Riverlane和加拿大量子计算公…

AI推介-多模态视觉语言模型VLMs论文速览(arXiv方向):2024.03.10-2024.03.15

论文目录~ 1.3D-VLA: A 3D Vision-Language-Action Generative World Model2.PosSAM: Panoptic Open-vocabulary Segment Anything3.Anomaly Detection by Adapting a pre-trained Vision Language Model4.Introducing Routing Functions to Vision-Language Parameter-Efficie…

机试:高精度乘法

问题描述 代码示例 #include <bits/stdc.h> using namespace std;int main(){char a[64],b[64];cin >> a >> b;int A[64],B[64],C[128];for(int i 0; i < strlen(a); i){A[i] a[i] - 0;}for(int i 0; i < strlen(b); i){B[i] b[i] - 0;}// 反转 …

【nnUNetv2实践】一、nnUNetv2安装

nnUNet是一个自适应的深度学习框架&#xff0c;专为医学图像分割任务设计。以下是关于nnUNet的详细解释和特点&#xff1a; 自适应框架&#xff1a;nnUNet能够根据具体的医学图像分割任务自动调整模型结构、训练参数等&#xff0c;从而避免了繁琐的手工调参过程。自动化流程&am…

详细分析Python装饰器(附Demo)

目录 前言1. 基本知识2. 无参装饰器3. 有参装饰器4. 多个装饰器 前言 装饰器类似Java的切点切面增强 推荐阅读&#xff1a; 详细分析Spring中的Around注解&#xff08;附Demo&#xff09;java框架 零基础从入门到精通的学习路线 附开源项目面经等&#xff08;超全&#xff0…

网络安全实训Day5

写在前面 昨天忘更新了......讲的内容不多&#xff0c;就一个NAT。 之前记的NAT的内容&#xff1a;blog.csdn.net/Yisitelz/article/details/131840119 网络安全实训-网络工程 NAT 公网地址与私网地址 公网地址 可以在互联网上被寻址&#xff0c;由运营商统一分配全球唯一的I…

git checkout不同分支时,为啥会把当前分支的修改内容也带到新分支里面?

git checkout不同分支时&#xff0c;为啥会把当前分支的修改内容也带到新分支里面&#xff1f; 当你在Git中从一个分支切换到另一个分支时&#xff0c;如果没有先将当前分支未提交的改动暂存或提交&#xff0c;这些改动会被带到新分支。这是因为Git的工作目录是共享的&#xf…