排序决战(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

相关文章

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

python连接本地SQL server详细图文教程

《python连接本地SQLserver详细图文教程》在数据分析领域,经常需要从数据库中获取数据进行分析和处理,下面:本文主要介绍python连接本地SQLserver的相关资料,文中通过代码... 目录一.设置本地账号1.新建用户2.开启双重验证3,开启TCP/IP本地服务二js.python连接实例1.

Nginx中配置HTTP/2协议的详细指南

《Nginx中配置HTTP/2协议的详细指南》HTTP/2是HTTP协议的下一代版本,旨在提高性能、减少延迟并优化现代网络环境中的通信效率,本文将为大家介绍Nginx配置HTTP/2协议想详细步骤,需... 目录一、HTTP/2 协议概述1.HTTP/22. HTTP/2 的核心特性3. HTTP/2 的优

Java图片压缩三种高效压缩方案详细解析

《Java图片压缩三种高效压缩方案详细解析》图片压缩通常涉及减少图片的尺寸缩放、调整图片的质量(针对JPEG、PNG等)、使用特定的算法来减少图片的数据量等,:本文主要介绍Java图片压缩三种高效... 目录一、基于OpenCV的智能尺寸压缩技术亮点:适用场景:二、JPEG质量参数压缩关键技术:压缩效果对比

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

Python基础文件操作方法超详细讲解(详解版)

《Python基础文件操作方法超详细讲解(详解版)》文件就是操作系统为用户或应用程序提供的一个读写硬盘的虚拟单位,文件的核心操作就是读和写,:本文主要介绍Python基础文件操作方法超详细讲解的相... 目录一、文件操作1. 文件打开与关闭1.1 打开文件1.2 关闭文件2. 访问模式及说明二、文件读写1.

Ubuntu中远程连接Mysql数据库的详细图文教程

《Ubuntu中远程连接Mysql数据库的详细图文教程》Ubuntu是一个以桌面应用为主的Linux发行版操作系统,这篇文章主要为大家详细介绍了Ubuntu中远程连接Mysql数据库的详细图文教程,有... 目录1、版本2、检查有没有mysql2.1 查询是否安装了Mysql包2.2 查看Mysql版本2.

Oracle数据库常见字段类型大全以及超详细解析

《Oracle数据库常见字段类型大全以及超详细解析》在Oracle数据库中查询特定表的字段个数通常需要使用SQL语句来完成,:本文主要介绍Oracle数据库常见字段类型大全以及超详细解析,文中通过... 目录前言一、字符类型(Character)1、CHAR:定长字符数据类型2、VARCHAR2:变长字符数