排序决战(2)堆排序——详细之详细

2024-02-13 06:40
文章标签 排序 详细 堆排序 决战

本文主要是介绍排序决战(2)堆排序——详细之详细,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上期讲完了希尔排序和插入排序后,我们的希尔排序成功胜出,这一期,我们继续,这次决战的是堆排序小姐

堆排序

概念:

堆排序即利用堆的思想来进行排序,总共分为两个步骤:

1.先把入的数据建堆成大堆(升序)/小堆(降序)

2.建成大堆/小堆后再进行取堆顶和堆底交换,进行依次向下调整,所以我们会用到向上调整向下调整算法,即可完成堆排序

如图所示:

 首先我们来讲解向上调整向下调整算法:

向上调整算法:

有如下的堆:9,8,6,5,6,1

 当我们想要插入堆数的时候,并且让堆保持堆的性质,我们应该如何操作

当我们把他调整为一个大堆/小堆,那么我们就会拿父亲节点和子节点比较,如果父亲节点大于/小于子节点的时候,则交换子节点,一直交换到父亲节点大于子节点的时候,便不进行交换了。

如图所示

 代码如下:

//向上调整 大堆
void AdjustUp(int* a,int child) {int parent = (child - 1) / 2;while (child>0){if (a[parent]<a[child]){Swap(&a[parent], &a[child]);child=parent;parent = (child - 1) / 2;}  else{break;}}
}

值得一提的是,我们堆的物理结构是数组,在数组中,我们应该如何得到他的父亲节点呢?

如图红色的是下标

看图可知我们则只需要让下标-1/2就好了

向下调整算法:

向下调整算法是整个堆排序的核心,

!但是最主要的一个==前提==就是根节点的左右子树都要是大堆或者都要是小堆,就根结点不满足,才可以去进行一个向下调整!

与向上调整算法类似,找子节点进行比较交换,一直到向下的子节点都符合堆的特性,不过在代码这里,我们有很多点要小心

void AdjustDown(int* a, int size, int parent) {//建大堆int child = parent*2+1;while (child<size){if (child+1<size&&a[child] < a[child + 1]){child++;//假设法}if (a[child]>a[parent]){Swap(&a[child],&a[parent]);parent = child;child = parent * 2 + 1;}else{break; }}		
}

在这里我们往下调整的时候,需要和左节点与右节点的最大一个换,那么,问题来了,我们怎么才能知道哪个是最大的呢?

这里,我们可以使用假设法,我们假设左节点/右节点最大,然后进行左节点和右节点的比较

知道父亲节点下标,求孩子节点下标
father = 0 :
左孩子 = father * 2 + 1 = 1;
右孩子 = father * 2 + 2 = 2;
father = 2:
左孩子 = father * 2 + 1 = 5;
右孩子 = father * 2 + 2 = 6;

 如果右节点比左节点大,那么便更新成为右节点为最大的节点,向下调整。如图:

 排序

开头讲了排序会用到以上两种算法,首先是建堆,然后进行与第n个交换向下调整,然后再和n--进行交换,循环进行

误区:

在这里,排升序的时候,如果按我们正常很流畅的逻辑来讲,我们自然的会去建小堆,但是事实真是如此吗?我们可以画个图来看看

由此可见,当要排升序的时候建小堆不是个明智的决定,而你觉得的,也不一定就是你觉得的,实践出真知,建小堆不仅关系会乱,而且效率也不高,所以,我们选择建大堆试试

如图所示:

如上图,红方框的已经按升序排好了,接下来一直排序到下标0就好了。

对照代码,来欣赏一下我们的堆排序小姐罢!

void Swap(int* p1, int* p2)
{int tmp = *p1;*p1 = *p2;*p2 = tmp;
}
//向上调整 大堆
void AdjustUp(int* a,int child) {int parent = (child - 1) / 2;while (child>0){if (a[parent]<a[child]){Swap(&a[parent], &a[child]);child=parent;parent = (child - 1) / 2;}  else{break;}}
}
void AdjustDown(int* a, int size, int parent) {//建大堆int child = parent*2+1;while (child<size){if (child+1<size&&a[child] < a[child + 1]){child++;//假设法}if (a[child]>a[parent]){Swap(&a[child],&a[parent]);parent = child;child = parent * 2 + 1;}else{break; }}		
}
void HeapSort(int* a, int size) {//建堆for (int i = 1; i < size; i++){AdjustUp(a, i);}int end = size - 1;for (int i = end; i >0; i--)//排序{Swap(&a[end], &a[0]);AdjustDown(a, end, 0);end--;}
}

时间复杂度分析

堆排序分作两个部分,

一是建堆,建堆要一个一个建,自然时间复杂度是On

二才是排序,调整这一块的话就是每次够把当前堆中最的数放到堆底来,然后每一个次大的数都需要向下调整O(log2N),数组中有N个数需要调整做排序,因而就是O(Nlog2N)

最后将两段代码整合一下,就是O(N + Nlog2N),取影响结果大的那一个就是O(Nlog2N),这也就是堆排序最终的时间复杂度

测试:

 可见,堆排序小姐速度并不算慢,比插入排序可快多了,但是已经略逊于我们上期的冠军,希尔 排序了

所以这一集,仍然是希尔排序胜利!

这篇关于排序决战(2)堆排序——详细之详细的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/704770

相关文章

Python pandas库自学超详细教程

《Pythonpandas库自学超详细教程》文章介绍了Pandas库的基本功能、安装方法及核心操作,涵盖数据导入(CSV/Excel等)、数据结构(Series、DataFrame)、数据清洗、转换... 目录一、什么是Pandas库(1)、Pandas 应用(2)、Pandas 功能(3)、数据结构二、安

Apache Ignite 与 Spring Boot 集成详细指南

《ApacheIgnite与SpringBoot集成详细指南》ApacheIgnite官方指南详解如何通过SpringBootStarter扩展实现自动配置,支持厚/轻客户端模式,简化Ign... 目录 一、背景:为什么需要这个集成? 二、两种集成方式(对应两种客户端模型) 三、方式一:自动配置 Thick

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

2025版mysql8.0.41 winx64 手动安装详细教程

《2025版mysql8.0.41winx64手动安装详细教程》本文指导Windows系统下MySQL安装配置,包含解压、设置环境变量、my.ini配置、初始化密码获取、服务安装与手动启动等步骤,... 目录一、下载安装包二、配置环境变量三、安装配置四、启动 mysql 服务,修改密码一、下载安装包安装地

在macOS上安装jenv管理JDK版本的详细步骤

《在macOS上安装jenv管理JDK版本的详细步骤》jEnv是一个命令行工具,正如它的官网所宣称的那样,它是来让你忘记怎么配置JAVA_HOME环境变量的神队友,:本文主要介绍在macOS上安装... 目录前言安装 jenv添加 JDK 版本到 jenv切换 JDK 版本总结前言China编程在开发 Java

Spring Boot Actuator应用监控与管理的详细步骤

《SpringBootActuator应用监控与管理的详细步骤》SpringBootActuator是SpringBoot的监控工具,提供健康检查、性能指标、日志管理等核心功能,支持自定义和扩展端... 目录一、 Spring Boot Actuator 概述二、 集成 Spring Boot Actuat

如何在Java Spring实现异步执行(详细篇)

《如何在JavaSpring实现异步执行(详细篇)》Spring框架通过@Async、Executor等实现异步执行,提升系统性能与响应速度,支持自定义线程池管理并发,本文给大家介绍如何在Sprin... 目录前言1. 使用 @Async 实现异步执行1.1 启用异步执行支持1.2 创建异步方法1.3 调用

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三

SpringBoot改造MCP服务器的详细说明(StreamableHTTP 类型)

《SpringBoot改造MCP服务器的详细说明(StreamableHTTP类型)》本文介绍了SpringBoot如何实现MCPStreamableHTTP服务器,并且使用CherryStudio... 目录SpringBoot改造MCP服务器(StreamableHTTP)1 项目说明2 使用说明2.1

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java