排序——堆排序

一、概念及结构

堆是一种特殊的树形数据结构,通常表现为一棵完全二叉树,是用二叉树的顺序存储方式来存储元素的。堆分为小堆和大堆

1.1 小堆

父结点小于子结点,但是父结点下面的两个子结点没大小区分。

1.2 大堆

父结点大于子结点,但是父结点下面的两个子结点没大小区分。

二、建堆(小堆示例)

2.1 向上调整建堆

适用于一边插入一边建堆的情况。插入节点后,找到它的父结点,让父结点与它进行比较,如果小于它则退出,如果大于则交换位置,重复上述操作直至循环结束或跳出循环。

cpp 复制代码
void AdjustUp(int* arr, int child)
{
	int parent = (child - 1) / 2;
	while (child > 0)
	{
		if (arr[child] < arr[parent])//改为 > 建大堆
		{
			Swap(&arr[child], &arr[parent]);
			child = parent;
			parent = (child - 1) / 2;
		}
		else
        {
            break;
        }
	}
}

注:Swap是交换位置函数。

cpp 复制代码
void Swap(int* p1, int* p2)
{
	int temp = *p1;
	*p1 = *p2;
	*p2 = temp;
}

2.2 向下调整建堆

适用于对父节点进行调整。先找到较小的那个孩子,让该孩子与它进行比较,如果孩子大于它则退出,如果小于则交换位置,重复上述操作直至循环结束或跳出循环。

cpp 复制代码
void AdjustDown(int* arr, int n, int parent)
{
	int child = (parent * 2) + 1;//先默认找到左孩子
	while (child < n)
	{
        //判断左孩子是否大于右孩子,child + 1 < n防止非法访问
		if (child + 1 < n && arr[child + 1] < arr[child])
			child++;

		if (arr[parent] > arr[child])
		{
			Swap(&arr[parent], &arr[child]);
			parent = child;
			child = (parent * 2) + 1;
		}
		else
        {
            break;
        }
	}
}

三、TOPK问题

TOPK问题:即求数据结合中前K个最大的元素或者最小的元素。一般来说数据量比较大。如果有一个数组,数组里有N个数据,我们要找最大的前K个数据,要怎么做呢?

3.1 法一:

先整体建立一个大堆,然后让根节点与该堆的最后一个节点交换,然后拿取该堆最后的结点,再让根结点向下调整。重复上述操作K次。

cpp 复制代码
void Heapsort(int* a, int n)
{
	int k = 0;
	scanf("%d", &k);
	for (int i = 0; i < n; i++)
		AdjustUp(a, i);

	while (k--)
	{
		Swap(&a[n - 1], &a[0]);
		AdjustDown(a, n - 1, 0);
		n--;
	}
}

3.2 法二:

先从数组中取前k个数建立一个小堆,然后遍历后面的元素,后与堆顶元素比较,如果大于则进行交换,然后向下调整。最后拿出该堆即可。

cpp 复制代码
void Heapsort(int* a, int n)
{
	int k = 0;
	scanf("%d", &k);
	for (int i = (k - 2) / 2; i >= 0; i--)
		AdjustDown(a, k, i);

	for (int i = k; i < n; i++)
	{
		if (a[i] > a[0])
		{
			Swap(&a[i], &a[0]);
			AdjustDown(a, k, 0);
		}
	}
}

(k - 2) / 2是为了拿到该堆中最后一个父节点的下标:子节点的下标求出父节点的下标的公式是 (i - 1) / 2 ,但是这里 k - 1 先当于是最后一个孩子节点的下标所以是 (k - 2) / 2。

相关推荐
笨笨饿3 分钟前
# 67_MCU的几大分区
数据结构·单片机·嵌入式硬件·算法·机器人·线性回归·个人开发
6Hzlia4 分钟前
【Hot 100 刷题计划】 LeetCode 230. 二叉搜索树中第 K 小的元素 | C++ 栈迭代中序遍历
c++·算法·leetcode
大熊背5 分钟前
ISP Pipeline中Lv实现方式探究之六--lv值计算再优化
网络·算法·自动曝光·lv
RTC老炮5 分钟前
WebRTC下FlexFEC算法架构及原理
网络·算法·音视频·webrtc
xin_nai6 分钟前
LeetCode热题100(Java)(2)双指针
算法·leetcode·职场和发展
七颗糖很甜8 分钟前
预警!超级厄尔尼诺即将登场:2026-2027年全球气候或迎“极端狂暴模式”
java·大数据·python·算法·github
Brilliantwxx9 分钟前
【C++】类与对象(下)
c++·笔记·算法
@insist12311 分钟前
信息安全工程师-密码学专题(上):密码学核心框架与基础体系梳理
算法·密码学·软考·信息安全工程师·软件水平考试
承渊政道12 分钟前
【动态规划算法】(从入门到精通:路径问题)
数据结构·c++·学习·算法·leetcode·macos·动态规划
进击的荆棘2 小时前
C++起始之路——哈希表的实现
数据结构·c++·散列表·哈希