【图解数据结构】堆排序

动图演示

在这里插入图片描述

实现步骤

  1. 把数组用堆的形式排列(小堆或者大堆)
  2. 对堆进行排序

Step1. 建堆

方法一:把第一个数就看成是一个堆,之后的数据依次当作push数据,再向上调整堆到合适位置。

  1. 如图,第一个数是70,已经是一个堆
    在这里插入图片描述
  2. 插入56,进行向上调整(此处认为建小堆)

在这里插入图片描述

  1. 插入30,再次调整堆,依次类推…

在这里插入图片描述

方法二:使用向下调整算法。

向下调整算法的前提左右子树都是堆

Q: 这里有一个问题?该从哪里开始调整。

因为叶子节点没有左右子树,叶子所在的子树不需要调整,所以要从倒数第一个非叶子节点的子树开始,即:最后一个结点的父亲。

在这里插入图片描述

Step2. 排序建堆

结论:
排升序,选大堆;排降序,选小堆。

如果排升序,选择小堆
每一次选走了堆顶元素,剩下的数字就会失去堆的结构,需要重新建堆。选择小堆不是不可以,但是时间复杂度是N*N,排序效率太低了。

在这里插入图片描述

如果排大堆

建大堆,可以选出最大的数;最大的数(堆顶)和最后一个数交换,相当于pop了堆顶后再次进行向下调整。向下调整算法的时间复杂度n个数 : N*logN

在这里插入图片描述

堆排序代码(排升序)

// 升序 
void HeapSort(int* a, int n)
{
	// 排升序,选大堆
	// 建堆:从倒数第一个非叶子节点开始向下调整
	// 为什么是end>=0 ?
	// 因为现在的堆顶元素是 a[0],需要再次进行向下调整,创建堆
	for (int i = (n - 1 - 1) / 2; i >= 0; --i)
	{
		AdjustDown(a, n, i);
	}

	// 升序,选大堆
	// 交换堆顶和最后一个元素后重新向下调整
	// 为什么是end>0 ?
	// 因为,只留下一个元素,已经是一个堆了,不用再调整了
	for (int end = n - 1; end > 0; --end)
	{
		// 交换
		Swap(&a[end], &a[0]);
		// 最后一个元素作为堆顶,再次向下调整
		AdjustDown(a, end, 0);
	}

}

向下调整

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;
		}
	}
}
  • 12
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 7
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

JoyCheung-

赏颗糖吃吧~~~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值