【数据结构】——建堆

一、堆是什么?

堆(heap)

堆(英语:heap)是计算机科学中一类特殊的数据结构的统称。堆通常是一个可以被看做一棵树的数组对象。堆总是一棵完全二叉树。

堆的性质:

  • 堆中某个节点的值总是不大于或不小于其父节点的值。

  • 堆中某个节点的值总是不大于或不小于其父节点的值。

堆的分类:

将根节点最大的堆叫做最大堆或大顶堆,根节点最小的堆叫做最小堆或小顶堆。常见的堆有二叉堆、斐波那契堆等。
请添加图片描述

二、向下调整算法

1.父子间的关系

请添加图片描述


2.向下调整算法

step 0:

请添加图片描述

step1

请添加图片描述

step2,3

请添加图片描述

注意事项
请添加图片描述
请添加图片描述

3.代码展示

void AdjustDown(int* a, int pos, int size)
{
	int parent = pos;
	int node = parent * 2 + 1;   //先默认左节点是两个结点中较大的一个
	while (parent * 2 + 1 < size) //调整过程是一个循环,当找到最后一层,即父节点已经没有子节点的时候,则说明调整完成
	{
		if (node + 1 < size &&  a[node] < a[node + 1] ) //比较左节点和右节点的大小,同时要注意没有右节点的情况
		{
			node = node + 1;   //如果右节点较大,则和parent交换的结点就要变成右节点
		}
		if (a[parent] < a[node])  //判断父结点和子节点的大小,如果子节点大,则交换
		{
			Swap(&a[parent], &a[node]);//交换

			//交换完成后,需要重新定位父节点和子节点
			parent = node;
			node = parent * 2 + 1;
		}
		else
		{
			break;   //如果父节点大,则说明已经整个二叉树已经调整完成了
		}
	}
}

三,建堆

从最后一个父亲节点开始调用,向下调整算法,直到 父亲节点为0截至

请添加图片描述

代码展示
int a[] = { 11,12,34,56,77,89 };
	int size = sizeof(a) / sizeof(a[0]);
	//建堆--大堆
	for(int i=(size-1-1)/2;i>=0;--i)
	{
			AdjustDown(a, i, size);
	}
全部代码展示
#include<stdio.h>

void Swap(int* a, int* b) //交换函数
{
	int tmp = *a;
	*a = *b;
	*b = tmp;
}

void AdjustDown(int* a, int pos, int size)
{
	int parent = pos;
	int node = parent * 2 + 1;   //先默认左节点是两个结点中较大的一个
	while (parent * 2 + 1 < size) //调整过程是一个循环,当找到最后一层,即父节点已经没有子节点的时候,则说明调整完成
	{
		if (node + 1 < size &&  a[node] < a[node + 1] ) //比较左节点和右节点的大小,同时要注意没有右节点的情况
		{
			node = node + 1;   //如果右节点较大,则和parent交换的结点就要变成右节点
		}
		if (a[parent] < a[node])  //判断父结点和子节点的大小,如果子节点大,则交换
		{
			Swap(&a[parent], &a[node]);//交换

			//交换完成后,需要重新定位父节点和子节点
			parent = node;
			node = parent * 2 + 1;
		}
		else
		{
			break;   //如果父节点大,则说明已经整个二叉树已经调整完成了
		}
	}
}
int main()
{
	int a[] = { 11,12,34,56,77,89 };
	int size = sizeof(a) / sizeof(a[0]);
	//建堆--大堆
	for(int i=(size-1-1)/2;i>=0;--i)
	{
			AdjustDown(a, i, size);
	}
	//打印出建好的堆
	for (int i = 0; i < size; i++)
	{
		printf("%d ", a[i]);
	}
	
	return 0;
}

效果展示
在这里插入图片描述

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值