C/C++ 堆排序

个人主页:仍有未知等待探索-CSDN博客

专题分栏:数据结构_仍有未知等待探索的博客-CSDN博客

                                                      欢迎大家来指教!

一、前言

今天要介绍的是堆排序。

首先什么是堆?简而言之,堆就是二叉树的数组形式,用数组来存储二叉树。

这个堆和C语言中讲的堆区是不同的两个概念,不要混淆。

二、堆排序

堆排序的核心就是构建一个特殊的二叉树,这个二叉树的特性是:其父节点大于等于(小于等于)其左右孩子结点。

故,最终创建的二叉树的根节点会是该二叉树的最大值(最小值)。

那怎么才能让整个数组中的数据有序呢?那我们就让数组中第一个数和最后一个数进行交换,然后以第一个数和倒数第二个数为端点,继续进行构建堆,然后选出次最小值。以此类推~~~

我以大根堆为例(从数组下标为1的地方开始存储):

1、堆的数据类型

typedef int HpDataType;
typedef struct Heap
{
	HpDataType* a;//进行存储堆中的数据
	int size;//该堆的有效数据个数
	int capacity;//该堆中的容量
}Hp;

2、堆的初始化

//初始化
void Heapinit(Hp* php)
{
	assert(php);//断言,如果php为空的话,报错
	php -> a = NULL;
	php -> size = 0;
	php -> capacity = 0;
}

3、堆的销毁

//堆的销毁
void HeapDestory(Hp* php)
{
	assert(php);
	free(php -> a);//释放掉开辟的空间
	php -> a = NULL;//防止php -> a为野指针
	php -> size = 0;
	php -> capacity = 0;
}

4、堆的创建

// 堆的创建
// k为要插入到堆里面的元素
void Heap_push(Hp* php, int k)
{
	assert(php); //断言
	if (php -> size == php -> capacity)// 如果容量不够,扩容
	{
		// 如果容量为0,则给个初始值,否则就二倍扩容
		int size = php -> capacity == 0 ? 4 : php -> capacity * 2;
		Hp* tmp = (Hp* ) realloc(php -> a, (size + 1) * sizeof(int));
		if (tmp == NULL) 
		{
			perror("realloc failed");
			exit(-1);// 扩容失败则结束程序
		}

		php -> a = tmp;
		php -> capacity = size;	
	}

	// 插入到堆的最后
	php -> a[++ php -> size] = k;

	// 向上调整,如果比父节点要大就进行交换
	up(php -> a, php -> size);
}

 5、向上调整

// 向上调整
// idx为要调整元素的下标
void up(int* a, int idx)
{
	assert(a);
	int child = idx;// 该结点下标
	int parent = child / 2;// 该结点的父节点,我是从下标为1开始存的

	while (child != 1)
	{
		if (a[parent] < a[child])
		{
			Swap(&a[parent], &a[child]);
		}
		else
		{
			// 不需要调整,就退出
			break;
		}

		// 继续向上找
		child = child / 2;
		parent = parent / 2;
	}
}

 6、删除堆的最后一个元素

该操作的主要作用是将该堆中最大的元素进行归位,并且进行调整除了最后一个元素的二叉树的顺序,使其仍为一个堆。

// 进行删除最后一个元素
void Heappop(Hp* php)
{
	assert(php);
	assert(php -> size > 0);
	
	Swap(&php -> a[1], &php -> a[php -> size]);
	php -> size --;

	int parent = 1;
	int child = 2 * parent;

	while (child <= php -> size)
	{
		// 假设左节点最大,如果右结点比左节点大,就child存右节点下标
		if (child + 1 <= php -> size && php -> a[child + 1] > php -> a[child])
		{
			child += 1;
		}
		// 判断孩子结点和父节点的值
		if (php -> a[parent] < php -> a[child])
		{
			Swap(&php -> a[parent], &php -> a[child]);
		}
		else 
		{
			// 不需要则退出
			break;
		}

		parent = child;
		child = 2 * parent;

	}
}

 7、取出堆顶元素

// 取出堆顶元素
HpDataType Heaptop(Hp* php)
{
	assert(php);
	assert(php -> size > 0);

	return php -> a[1];
}

三、总代码及其运行结果

#define _CRT_SECURE_NO_WARNINGS  1

// 大根堆

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>

typedef int HpDataType;
typedef struct Heap
{
	HpDataType* a;// 进行存储堆中的数据
	int size;// 该堆的有效数据个数
	int capacity;// 该堆中的容量
}Hp;

void Heapinit(Hp* php);// 初始化
void HeapDestory(Hp* php);// 堆的销毁
void Heap_push(Hp* php, int k);// 堆的创建
void Swap(int* a, int* b);// 交换
void up(int* a, int idx);// 向上调整
HpDataType Heaptop(Hp* php);// 取堆顶元素
void Heappop(Hp* php);// 删除堆的最后一个元素

int main()
{
	int a[] = {1, 2, 3, 4, 5, 6, 7};

	Hp hp;

	Heapinit(&hp);

	int len = sizeof(a) / sizeof(a[0]);
	for (int i = 0; i < len; i++ )
	{
		Heap_push(&hp, a[i]);
	}

	printf("初始堆:\n");
	for (int i = 1; i <= len; i ++ )
	{
		printf("%d ", hp.a[i]);
	}

	printf("\n有序序列:\n");

	while (hp.size > 0)
	{
		printf("%d ", Heaptop(&hp));

		Heappop(&hp);
	}

	HeapDestory(&hp);

	return 0;
}

// 初始化
void Heapinit(Hp* php)
{
	assert(php);// 断言,如果php为空的话,报错
	php -> a = NULL;
	php -> size = 0;
	php -> capacity = 0;
}

// 堆的销毁
void HeapDestory(Hp* php)
{
	assert(php);
	free(php -> a);// 释放掉开辟的空间
	php -> a = NULL;// 防止php -> a为野指针
	php -> size = 0;
	php -> capacity = 0;
}

void Swap(int* a, int* b)
{
	int tmp = *a;
	*a = *b;
	*b = tmp;
}

// 向上调整
// idx为要调整元素的下标
void up(int* a, int idx)
{
	assert(a);
	int child = idx;// 该结点下标
	int parent = child / 2;// 该结点的父节点,我是从下标为1开始存的

	while (child != 1)
	{
		if (a[parent] < a[child])
		{
			Swap(&a[parent], &a[child]);
		}
		else
		{
			// 不需要调整,就退出
			break;
		}

		// 继续向上找
		child = child / 2;
		parent = parent / 2;
	}
}

// 堆的创建
// k为要插入到堆里面的元素
void Heap_push(Hp* php, int k)
{
	assert(php); //断言
	if (php -> size == php -> capacity)// 如果容量不够,扩容
	{
		// 如果容量为0,则给个初始值,否则就二倍扩容
		int size = php -> capacity == 0 ? 4 : php -> capacity * 2;
		Hp* tmp = (Hp* ) realloc(php -> a, (size + 1) * sizeof(int));
		if (tmp == NULL) 
		{
			perror("realloc failed");
			exit(-1);// 扩容失败则结束程序
		}

		php -> a = tmp;
		php -> capacity = size;	
	}

	// 插入到堆的最后
	php -> a[++ php -> size] = k;

	// 向上调整,如果比父节点要大就进行交换
	up(php -> a, php -> size);
}

// 取出堆顶元素
HpDataType Heaptop(Hp* php)
{
	assert(php);
	assert(php -> size > 0);

	return php -> a[1];
}

// 进行删除最后一个元素
void Heappop(Hp* php)
{
	assert(php);
	assert(php -> size > 0);
	
	Swap(&php -> a[1], &php -> a[php -> size]);
	php -> size --;

	int parent = 1;
	int child = 2 * parent;

	while (child <= php -> size)
	{
		// 假设左节点最大,如果右结点比左节点大,就child存右节点下标
		if (child + 1 <= php -> size && php -> a[child + 1] > php -> a[child])
		{
			child += 1;
		}
		// 判断孩子结点和父节点的值
		if (php -> a[parent] < php -> a[child])
		{
			Swap(&php -> a[parent], &php -> a[child]);
		}
		else 
		{
			// 不需要则退出
			break;
		}

		parent = child;
		child = 2 * parent;

	}
}

 

谢谢大家! 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/311638.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

c++最值查找

目录 min和max函数 min_element和max_element 例 nth_element函数 例 例题 题目描述 输入描述 输出描述 解 min和max函数 只能传入两个值或一个列表 时间复杂度为O(1),数组O(n)&#xff0c;n为元素个数 min_element和max_element min_element(st,ed)返回地址[st,…

企业泛域名SSL证书

SSL数字证书是一种电子证书&#xff0c;它由CA认证机构颁发&#xff0c;提供了加密连接和身份验证的功能。SSL数字证书广泛应用于各种在线服务&#xff0c;如网页浏览、电子邮件、在线银行等。它能够提供安全的连接&#xff0c;保护用户的个人信息和交易数据不被窃取或篡改。企…

如果你还忍受校园网,那么一定要看它!随身WiFi哪个品牌最靠谱 ?高人气随身WiFi第一名

很多学生党反映校园网价格太贵&#xff0c;贵就算了&#xff0c;还非常不好用&#xff0c;真的很泪崩。 对于学生党来说&#xff0c;一个性价比极高的随身WiFi才是正正好。 但是很多学生党对随身WiFi并不了解&#xff0c;那么下面我就回答几个学生党最关心的问题。一、价格合…

JVM类加载器打破双亲委派机制

欢迎大家关注我的微信公众号&#xff1a; 传送门&#xff1a;从JDK源码级别剖析JVM类加载器 目录 打破双亲委派机制 Tomcat打破双亲委派机制 Tomcat自定义加载器详解 模拟实现Tomcat打破双亲委派 上一篇文章讲到了jvm类加载器的双亲委派机制&#xff0c;本文就来讲…

探索 hasOwnProperty:处理对象属性的关键(下)

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

IntelliJ IDEA Java 连接 mysql 配置(附完整 demo)

下载 MySQL 驱动 从MySQL官网下载JDBC驱动的步骤如下&#xff1a; 1&#xff09;访问MySQL的官方网站&#xff1a;MySQL 2&#xff09;点击页面上方的"DOWNLOADS"菜单&#xff1b; 3&#xff09;在下载页面&#xff0c;找到"MySQL Community (GPL) Downloads…

人工智能推动供应链革命的成功

人工智能推动供应链革命的成功 目录 人工智能推动供应链革命的成功一、供应链管理不断变化的面貌二、拥挤的解决方案景观三、踏上人工智能驱动的转型1. 价值创造识别、战略和路线图2. 目标解决方案设计和供应商选择3. 实施与系统集成4. 变革管理、能力建设和全面价值获取 新技术…

MySQL-索引回顾

索引是面试高频问答题&#xff0c;参考百度/CSDN/尚硅谷/黑马程序员/阿里云开发者社区&#xff0c;决定将索引知识回顾一下&#xff0c;忘记时&#xff0c;点开即可&#xff0c;时刻保持更新&#xff0c;事不宜迟&#xff0c;即刻享用。 索引概述 索引&#xff08;index&#…

Python算法例35 丑数Ⅰ

1. 问题描述 丑数的定义是&#xff0c;只包含质因子2、3、5的正整数&#xff0c;例如6、8就是丑数&#xff0c;但14不是丑数&#xff0c;因为它包含了质因子7&#xff0c;本例将检测一个整数是不是丑数。 2. 问题示例 给出num8&#xff0c;返回True&#xff1b;给出num14&am…

【LeetCode】第二高的薪水(数据库)

目录 题目&#xff1a; 方法一 验证一&#xff1a; 验证二&#xff1a; 方法二 验证一&#xff1a; 验证二&#xff1a; 方法三 验证一&#xff1a; 验证二&#xff1a; 题目&#xff1a; 方法一 SELECT DISTINCT Salary AS SecondHighestSalary FROM Employee O…

图片双线性插值原理解析与代码 Python

一、原理解析 图片插值是图片操作中最常用的操作之一。为了详细解析其原理&#xff0c;本文以 33 图片插值到 55 图片为例进行解析。如上图左边蓝色方框是 55 的目标图片&#xff0c;右边红色方框是 33 的源图片。上图中&#xff0c;蓝/红色方框是图片&#xff0c;图片中的蓝/红…

20个很棒的SEO博客和网站

搜索引擎优化 &#xff08;SEO&#xff09; 标准在不断变化。与社交媒体、电子邮件营销和人工智能 &#xff08;AI&#xff09; 等新兴技术一样&#xff0c;搜索引擎正在改进它们每天向用户提供结果的方式。 为此&#xff0c;他们专注于本地化、页面权限、点击率&#xff0c;甚…

k8s中top指令使用前提:正确安装metrics-server

参考引用项目&#xff1a;https://www.cnblogs.com/lfl17718347843/p/14283796.html Kubernetes Metrics Server 是 Cluster 的核心监控数据的聚合器&#xff0c;kubeadm 默认是不部署的。 确认metrics-server能否被使用的三个前提&#xff08;验证以及修改方法https://cnblogs…

peropure·AI:开创智能助手新纪元

在当今数字化时代&#xff0c;人工智能&#xff08;AI&#xff09;技术正迅速发展&#xff0c;并深刻地改变着我们的生活方式。在这个背景下&#xff0c;peropureAI崭露头角&#xff0c;作为一款基于先进AI大模型的对话式个人助理工具&#xff0c;它在多个场景中展现出卓越的实…

机器学习之心荣获2023博客之星TOP50 | 感谢CSDN

机器学习之心荣获2023博客之星TOP50 | 感谢CSDN 2023年&#xff0c;是极其不平凡的一年&#xff01;感谢CSDN平台&#xff01;感谢所有支持我的前辈、朋友和同学&#xff01;2024年&#xff0c;新的一年&#xff0c;继续努力&#xff0c;持续写作&#xff01;

MongoDB调优

三大导致 MongoDB 性能不佳的原因 1&#xff09;慢查询2&#xff09;阻塞等待3&#xff09;硬件资源不足1,2通常是因为模型/索引设计不佳导致的。排查思路&#xff1a;按1-2-3依次排查。 影响 MongoDB 性能的因素 MongoDB 性能监控工具 Free Monitoring 从版本 4.0 开始&am…

离散数学3

补变元 解释&#xff1a;它是以反^作为一组一组的&#xff0c;因此&#xff0c;对于P反^Q来说&#xff0c;P是一组&#xff0c;Q是一组&#xff0c;又有以下&#xff1a;对缺少变元的项要补齐&#xff0c;P缺少Q&#xff0c;Q缺少P。因此&#xff0c;补齐。 用分配律展开 解释&…

全网第一篇教你怎么总结多线程知识

于「全景图」&#xff0c;我之前也有一直在构建&#xff0c;可是因为知识储备不够&#xff0c;确实很难构建出来。稍微了解过并发领域知识的人都知道&#xff0c;里面的知识点、概念多而散&#xff1a;线程安全、锁、同步、异步、阻塞、非阻塞、死锁、队列(为什么并发要跟队列扯…

OpenCV4.x(C++)人脸检测(眼睛、侧脸、正脸)

一、前言 OpenCV是一款广泛使用的计算机视觉库&#xff0c;提供了许多强大的功能&#xff0c;包括人脸检测和识别。人脸分类器是OpenCV中用于人脸检测的关键工具之一&#xff0c;能够快速准确地检测出图像中的人脸。 本文将介绍如何使用OpenCV自带的人脸分类器&#xff0c;并…

SpringBoot中使用SpringEvent业务解耦神器实现监听发布事件同步异步执行任务

场景 SpringBoot中使用单例模式ScheduledExecutorService实现异步多线程任务(若依源码学习)&#xff1a; SpringBoot中使用单例模式ScheduledExecutorService实现异步多线程任务(若依源码学习)-CSDN博客 设计模式-观察者模式在Java中的使用示例-环境监测系统&#xff1a; 设…