数据结构第九天(堆排序)

目录

前言

概述

源码:

主函数:

运行结果:

其他



前言

哈哈,这个堆排序算法很久之前就已经敲过一遍了,时间一久,思路有点淡忘。今天重新看过一遍之后,又亲自撸代码,幸运的是,代码运行一次就成功了,没有任何逻辑错误而且结果也达到了预期效果。

在最后,与大家共勉:你所走的每一步路,都算数。

概述

堆排序(Heap Sort)是一种基于比较的排序算法,使用二叉堆(Binary Heap)数据结构来帮助实现其排序过程。二叉堆可以被视为一颗完全二叉树,其中每个节点的值都不大于(或不小于)其子节点的值,这样的二叉堆分别称为最大堆(Max Heap)和最小堆(Min Heap)。堆排序主要包括两个步骤:建堆(构造初始堆)和调整堆。

1. 建堆(构造初始堆)

首先,将待排序的序列构造成一个最大堆,即所有父节点的值都大于或等于子节点的值。这一步的目的是选出最大的元素(位于根节点),准备将它与序列的末尾元素交换。

建堆的过程从最后一个非叶子节点开始,向前逐个对每个父节点进行调整,确保每个父节点的值都大于其子节点的值。最后一个非叶子节点的位置可以通过序列长度计算得出,设序列长度为n,则最后一个非叶子节点的位置是n/2 - 1(假定序列的起始索引为0)。

2. 调整堆

将最大堆的根节点(即当前最大值)与最后一个元素交换,然后减少堆的大小,对新的根节点执行下沉操作,以重新满足最大堆的性质。重复这个过程,直到堆的大小为1,排序完成。

下沉操作指的是将新的根节点值与其子节点中较大者交换,直到该节点值大于其子节点或已经到达叶子节点。

堆排序的算法步骤

  1. 建立最大堆:从最后一个非叶子节点开始,向上进行调整,确保每个父节点都大于其子节点。
  2. 排序
    • 将根节点(最大值)与最后一个元素交换,此时序列的最后部分已经是排序好的。
    • 减少堆的大小(排除已排序的部分),对新的根节点进行下沉操作,以维护最大堆的性质。
    • 重复上述过程,直到堆的大小为1。

时间复杂度

  • 最好、最坏、平均情况下的时间复杂度均为O(n log n)。
  • 堆排序的空间复杂度为O(1),因为它是原地排序算法。

特点

  • 不稳定排序:相同的元素可能会因为堆调整过程而改变其相对位置。
  • 原地排序:不需要额外的存储空间。

堆排序是一种高效的排序算法,尤其适用于数据量大的情况。由于其在最坏情况下也能保持O(n log n)的时间复杂度,因此是一种非常可靠的排序方法。

源码:

void heapAdjust(int* dest,  unsigned int dataCnt)
{
	unsigned int head = dataCnt;
	head /= 2;
	if (head * 2 + 1 == dataCnt)
	{
		if (*(dest + head * 2) > *(dest + head * 2 - 1))
		{
			if (*(dest + head - 1) > *(dest + head * 2 - 1))
			{
				swap(*(dest + head - 1), *(dest + head * 2 - 1));
			}
		}
		else if (*(dest + head - 1) > *(dest + head * 2))
		{
			swap(*(dest + head - 1), *(dest + head * 2));
		}
	}
	else{
		if (*(dest + head -1) > *(dest + head * 2 - 1))
		{
			swap(*(dest + head - 1), *(dest + head * 2-1));
		}
	}
	
	for (int i = dataCnt/2-1; i > 0; --i)
	{
		head = i;
		if (*(dest + head * 2) > *(dest + head * 2 - 1))
		{
			if (*(dest + head - 1) > *(dest + head * 2 - 1))
			{
				swap(*(dest + head - 1), *(dest + head * 2 - 1));
			}
		}
		else if (*(dest + head - 1) > *(dest + head * 2))
		{
			swap(*(dest + head - 1), *(dest + head * 2));
		}
		
	}
	
}
void sortByHeapSort(int* dest, const unsigned int dataCnt)
{
	for (int i = 0; i < dataCnt; ++i)
	{
		heapAdjust(dest + i, dataCnt - i);
	}
}

主函数:

#include<stdio.h>
#include<iostream>
using namespace std;
#include"dataStructAPI.h"
#include"sort.h"
#include<windows.h>
int main()
{

int array[16] = { 0 };
    numberProducer.getFilledArray(array,16);
    cout << "  原 始 数 据   :";
    numberProducer.showArray(array,16);


    sortByHeapSort(array, 16);
    cout << "   堆 排 序 后  :";
    numberProducer.showArray(array, 16);


    sortBySelectSort(array, 16);
    cout << "选 择 排 序 后  :";
    numberProducer.showArray(array, 16);

    sortByQuickSort(array, 16);
    cout << "快 速 排 序 后  :";
    numberProducer.showArray(array, 16);

    sortByBubbleSort(array, 16);
    cout << "冒 泡 排 序 后  :";
    numberProducer.showArray(array, 16);

    sortByShellInsert(array, 16, 3);
    cout << "希尔插入排序后  :";
    numberProducer.showArray(array, 16);

    sortByBinarySearchInsert(array, 16);
    cout << "折半插入排序后  :";
    numberProducer.showArray(array, 16);

    sortByDirectInsert(array, 16);
    cout << "直接插入排序后  :";
    numberProducer.showArray(array, 16);
    
    system("pause");
    return 0;
}

运行结果:

其他

数据结构第一天(生成1000以内的随机数自动填充数组)

数据结构第二天(直接插入排序/内存申请/指针操作)

数据结构第三天(折半插入排序)

数据结构第四天(希尔排序)

数据结构第五天(冒泡排序)

数据结构第六天(快速排序)

数据结构第七天(简单选择排序)

数据结构第八天(归并排序)

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

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

相关文章

【RL】Bellman Equation (贝尔曼等式)

Lecture2: Bellman Equation State value 考虑grid-world的单步过程&#xff1a; S t → A t R t 1 , S t 1 S_t \xrightarrow[]{A_t} R_{t 1}, S_{t 1} St​At​ ​Rt1​,St1​ t t t, t 1 t 1 t1&#xff1a;时间戳 S t S_t St​&#xff1a;时间 t t t时所处的sta…

C++ | vector二维数组的初始化与行、列数的获取

如果直接使用vector<int,vector<int> > v;创建二维数组&#xff0c;那么就会得到一个空的容器&#xff0c;这样再通过push_back赋值是非常麻烦的。 初始化二维数组 在此介绍二维数组初始化的一般操作。 首先看一维数组的初始化示例&#xff1a; 定义一个长度为n&a…

拿捏循环链表

目录&#xff1a; 一&#xff1a;单链表&#xff08;不带头单向不循环&#xff09;与循环链表&#xff08;带头双向循环&#xff09;区别 二&#xff1a;循环链表初始化 三&#xff1a;循环链表头插 四&#xff1a;循环链表尾插 五&#xff1a;循环链表头删 六&#xff1…

MessageBox好用吗?

MessageBox作为一种能够对接多系统平台的工具&#xff0c;在数字营销领域具有很大的实用性和价值。它提供了实时的数据同步、个性化的营销策略、用户互动功能等多种功能&#xff0c;可以帮助企业实现更精准、高效的营销活动。具体来说&#xff0c;MessageBox的优点包括&#xf…

Laykefu客服系统后台登录绕过

【产品介绍】 Laykefu 是一款基于workermangatawayworkerthinkphp5搭建的全功能webim客服系统&#xff0c;旨在帮助企业有效管理和提供优质的客户服务 【漏洞介绍】 请求头中Cookie中的”user_name“不为空时即可绕过登录系统后台&#xff0c;恶意攻击者可利用此漏洞获得后台…

基于摄像头的虹膜识别技术

随着苹果公司的指纹识别TouchID的推广流行&#xff0c;三星等公司的积极跟进&#xff0c;生物识别技术正被移动设备厂商所重视。 虹膜是什么&#xff1f; 人的眼睛由巩膜、虹膜、瞳孔三部分构成。巩膜即眼球外围的白色部分&#xff0c;约占总面积的30%&#xff1b;眼睛中心为瞳…

【React】如何使antd禁用状态的表单输入组件响应点击事件?

最近遇到一个需求&#xff0c;需要在<Input.textarea>组件中&#xff0c;设置属性disabled为true&#xff0c;使textarea响应点击事件&#xff0c;但直接绑定onClick并不会在禁用状态下被响应。 解决方法1 之后尝试了很多方法&#xff0c;比如设置csspointer-events:no…

日本失去的三十年:去杠杆用了14年

去年以来&#xff0c;日股在日本央行转鹰预期、基本面改善和一系列监管新规的催化下高歌猛进&#xff0c;日经指数已经逼近90年代资产泡沫时期的高位。今年迄今累计上涨8.51%&#xff0c;领跑全球&#xff0c;“失落的三十年”似乎已经远去。 日本因何走向衰退&#xff1f;“失…

MPLS VPN功能组件(2)

MP-BGP 采用地址族(Address Family)来区分不同的网络层协议,以便正确处理VPN-IPv4路由 传统的BGP-4(RFC1771)只能管理IPv4的路由信息,无法正确处理地址空间重叠的VPN的路由。 为了正确处理VPN路由,VPN使用RFC2858(Multiprotocol Extensions for BGP-4)中规定的MP-BG…

云计算 - 弹性计算技术全解与实践

一、引言 在过去的十年里&#xff0c;云计算从一个前沿概念发展为企业和开发者的必备工具。传统的计算模型通常局限于单一的、物理的位置和有限的资源&#xff0c;而云计算则通过分布式的资源和服务&#xff0c;为计算能力带来了前所未有的"弹性"。 弹性&#xff1a;…

TryHackMe-Vulnerability Capstone练习

本文相关的TryHackMe实验房间链接&#xff1a;TryHackMe | Vulnerability Capstone 先nmap扫一下 接下来我们访问一下 接下来我们searchsploit找一下漏洞 searchsploit Fuel CMS 执行漏洞exp&#xff08;此处使用TryHackMe中的box&#xff09; 如果使用本地机需要下载exp&am…

算法练习-删除二叉搜索树中的节点(思路+流程图+代码)

难度参考 难度&#xff1a;中等 分类&#xff1a;二叉树 难度与分类由我所参与的培训课程提供&#xff0c;但需要注意的是&#xff0c;难度与分类仅供参考。且所在课程未提供测试平台&#xff0c;故实现代码主要为自行测试的那种&#xff0c;以下内容均为个人笔记&#xff0c;旨…

制作离线版element ui文档

链接&#xff1a;https://pan.baidu.com/s/1k5bsCK9WUlZobhFBLItw1g?pwdgeyk 提取码&#xff1a;geyk --来自百度网盘超级会员V4的分享 https://github.com/ElemeFE/element 克隆官方代码 使用nvm切换node版本&#xff0c;推荐使用14.0.0 http://doc.xutongbao.top/doc/#/zh…

Prompt Engineering实战-构建“哄哄模拟器”

目录 一 背景 二 “哄哄模拟器”的Prompt Prompt 的典型构成 三 操作步骤 3.1 创建对话 3.2 游戏测试 一 背景 前几天《AI 大模型全栈工程师》第二节课讲了“Prompt Engineering&#xff0c;提示工程”&#xff0c;里面提到一些prompt相关的技巧&#xff0c;原则&#xf…

浅谈分布式CAP定律、BASE理论

第一节 分布式架构设计理论与Zookeeper环境搭建 1. 分布式架构设计理论 学习Zookeeper之前,我们需要掌握一些分布式系统基础知识&#xff1a;了解分布式系统的概念、原理。 配置管理 域名服务 分布式同步 发布订阅 1. 分布式架构介绍 1.1 什么是分布式 《分布式系统原理和…

构建异步高并发服务器:Netty与Spring Boot的完美结合

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 ChatGPT体验地址 文章目录 前言IONetty1. 引入依赖2. 服务端4. 客户端结果 总结引导类-Bootstarp和ServerBootstrap连接-NioSocketChannel事件组-EventLoopGroup和NioEventLoopGroup 送书…

COMSOL接触(高度非线性)仿真常见报错及解决方法总结

前言 由于COMSOL采用隐式求解器&#xff0c;相较于使用显式求解器的Dyna、Abaqus等软件。要在COMSOL中实现结构接触这一高度非线性问题难度较大&#xff0c;报错时有发生。究其原因&#xff0c;是当物体之间相互接触时&#xff0c;物体受到的应力、运动路径会发生突变&#xff…

LabVIEW高精度微小电容测量

LabVIEW高精度微小电容测量 在电子工程和科研领域&#xff0c;精确测量微小电容值是一项有一定要求的任务&#xff0c;尤其在涉及到高精度和低成本时。设计了一种基于LabVIEW高精度微小电容测量系统&#xff0c;旨在提供一个既经济又高效的解决方案。 该系统的核心在于使用FD…

C语言中的内存函数你知道多少呢?

目录 ​编辑 1. memcpy的使用和模拟实现 1.1函数介绍 ​编辑 1.2函数的使用 1.3模拟实现 2. memmove的使用和模拟实现 2.1函数介绍 2.2函数的使用 2.3模拟实现 3. memset函数的使用 3.1函数介绍 3.2函数的使用 ​编辑 4. memcmp函数的使用 4.1函数介绍 4.2函数…

Python循环语句——range语句

一、引言 在Python编程中&#xff0c;range函数是一个内置函数&#xff0c;用于生成一个不可变的数字序列。它常被用于循环结构&#xff0c;如for循环&#xff0c;来遍历一系列的数字。尽管其使用非常基础&#xff0c;但range的强大之处在于其提供了灵活性&#xff0c;可以创建…
最新文章