堆排序

    <div id="post_detail">

# 预备知识

    堆是一种特殊的树形数据结构,即完全二叉树。堆分为大根堆和小根堆,大根堆为根节点的值大于两个子节点的值;小根堆为根节点的值小于两个子节点的值,同时根节点的两个子树也分别是一个堆。

                                                                     imageimage

# 基本思路

  • 步骤一:建立大根堆--将n个元素组成的无序序列构建一个大根堆,
  • 步骤二:交换堆元素--交换堆尾元素和堆首元素,使堆尾元素为最大元素;
  • 步骤三:重建大根堆--将前n-1个元素组成的无序序列调整为大根堆

    重复执行步骤二和步骤三,直到整个序列有序。

# 图示说明

  • 步骤一:建立大根堆

① 无序序列建立完全二叉树

image

② 从最后一个叶子节点开始,从左到右,从下到上调整,将完全二叉树调整为大根堆

a.找到第1个非叶子节点6,由于6的右子节点9比6大,所以交换6和9。交换后,符合大根堆的结构。

image

c.找到第2个非叶子节点4,由于的4左子节点9比4大,所以交换4和9。交换后不符合大根堆的结构,继续从右到左,从下到上调整。

image

image

 

  • 步骤二:交换堆元素(交换堆首和堆尾元素--获得最大元素)

image

  • 步骤三:重建大根堆(前n-1个元素)

image

  • 重复执行步骤二和步骤三,直到整个序列有序

image

# C++代码

#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;


void Max_Heaplfy(int* a, int length, int index)//此函数保证根节点值最大,此函数只处理了以比
										//其大的孩子为根的子树,因为建堆是自下至上的,下面的已经符合最大堆
{
	int left = 2 * index + 1;
	int right = 2 * index + 2;
	int largest = index;
	if (left <= length - 1 && a[left] > a[index])//子节点不能大于总节点数
		largest = left;
	if (right <= length-1 && a[right] > a[largest])
		largest = right;
	if (largest != index)
	{
		swap(a[index],a[largest]);
		Max_Heaplfy(a, length, largest); //递归对以a[largest]为根的子树进行调整,保证a[largest]最大
	}
}

void Build_Max_Heap(int* a, int length)//构建堆
{
	for (int i = (length) / 2 -1; i >= 0; i--)//堆是一个完全二叉树,二叉树的叶子节点为(n/2.....n),非叶子节点为(0-n/2-1)
	{
		Max_Heaplfy(a, length , i);//对每个非叶子节点调整以其为根的子树
	}

}

void main1()
{
	int a[] = { 4, 2, 1, 8, 0, 3 };
	int len = sizeof(a) / sizeof(a[0]);
	
	Build_Max_Heap(a, len);
	
	for (int i = len-1 ; i >= 1; i--)
	{
		swap(a[i], a[0]);//升序排列
		Max_Heaplfy(a, i, 0);//对剩下的维护堆
	}

	for (int j = 0; j < len; j++)
	{
		cout << a[j] << " ";
	}
	system("pause");
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值