堆排及时间复杂度分析

2024-02-08 19:44
文章标签 分析 复杂度 时间 堆排

本文主要是介绍堆排及时间复杂度分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

箴言:

初始阶段,不需要去纠结那一种更优美,非要找出那一种是最好的,其实能解决问题的就是好办法。

一,常见排序时间复杂度

冒泡快排归并堆排桶排
时间O(n^2)O(nlogn)O(nlogn)O(nlogn)kn
空间O(1)O(1)O(nlogn)O(1)kn

二,堆排

前情提要:

堆属于完全树,完全树可以理解为一个数组。如果不是完全树,就没办法和数组等价,就不会有下面这种父级和子级之间的关系。

已知父级下标i
左孩子下标: 2*i+1
右孩子下标: 2*i+2
已知孩子结点j(无论左还是右)
父级下标 (j-1)/2

堆排序过程:

堆排序分成两个阶段,第一个阶段从由无序数组建立一个大/小根堆,第二个阶段在大/小根堆的基础上调整,形成有序数组。

从无序数组到大根堆:

对于数组中每一个元素,我们需要将其和其父级做对比,若比父级大,则进行交换,直到最顶层为止。

代码:(其实找父亲的时候可以不区分左右减一除二即可,我这里就不改了)

    public static void builddui(int[] arr) {for (int i = 0; i < arr.length; i++) {int j = i;int p = 0;if (j % 2 == 1) {//左孩子p = (j - 1) / 2;} else {p = (j - 2) / 2;//右孩子}while (p >= 0 && arr[p] < arr[j]) {int t = arr[p];//交换位置arr[p] = arr[j];arr[j] = t;j = p;p = (j - 1) / 2;}}}

从大根堆到有序序列:

最后一个位置和堆顶交换,将交换之后的堆顶下沉到正确的位置。然后堆顶和倒数第二个交换,堆顶下沉到正确的位置,直到剩下一个为止。这是一个堆顶元素不断下沉的过程。

代码:(r表示的是最后一个的索引位置)

    public static void weichidui(int[] arr, int r) {int t = arr[r];arr[r] = arr[0];arr[0] = t;int cur = 0;//当前下标while (2 * cur + 1 < r) {int index = 2 * cur + 1;int maxv = arr[index];if (2 * cur + 2 < r && arr[index] < arr[2 * cur + 2]) {index = 2 * cur + 2;maxv = arr[2 * cur + 2];}if (maxv > arr[cur]) {int tmp = arr[cur];arr[cur] = arr[index];arr[index] = tmp;}cur = index;}}

时间复杂度分析:

上述两个阶段分别分析: 从无序序列到建成大顶堆: 已知数组中数量为n,每正确插入一个元素,时间复杂度为logn(因为树的深度为logn),因为插入n个元素,时间复杂度为nlogn。

从大顶堆到有序序列:每次首尾交换之后都需要将堆顶元素下沉到正确的位置,时间复杂度为logn(因为树的深度为logn,比较交换次数其实是小于logn的,但是理解为logn就行),需要下沉n次,所以时间复杂度是nlogn。

ABOVE ALL,堆排时间复杂度为2nlogn,也就是O(nlogn),一切操作都是在原数组上进行的操作,所以空间复杂度为O(1)。

堆排序是一个完美的排序方式,无论时间或者空间,数据量小的时候差距不明显,数据量越大,优势就会越明显。

代码:

数组:[34,56,23,33,5,46,4,57,6,76,34,42,634,6,536,3,3423,3,1,5,537,3,57,3563,4,65,764,4]

import java.util.Arrays;/*** @Author YuLing* @Date 2024-02-07 19:14* @Description:* @Version 1.0*/
public class dui {public static void main(String[] args) {int[] arr = new int[]{34,56,23,33,5,46,4,57,6,76,34,42,634,6,536,3,3423,3,1,5,537,3,57,3563,4,65,764,4};builddui(arr);System.out.println(Arrays.toString(arr));for (int i = 0; i < arr.length; i++) {weichidui(arr,  arr.length - 1 - i);}System.out.println(Arrays.toString(arr));}public static void builddui(int[] arr) {for (int i = 0; i < arr.length; i++) {int j = i;int p = 0;if (j % 2 == 1) {//左孩子p = (j - 1) / 2;} else {p = (j - 2) / 2;//右孩子}while (p >= 0 && arr[p] < arr[j]) {int t = arr[p];//交换位置arr[p] = arr[j];arr[j] = t;j = p;p = (j - 1) / 2;}}}public static void weichidui(int[] arr, int r) {int t = arr[r];arr[r] = arr[0];arr[0] = t;int cur = 0;//当前下标while (2 * cur + 1 < r) {int index = 2 * cur + 1;int maxv = arr[index];if (2 * cur + 2 < r && arr[index] < arr[2 * cur + 2]) {index = 2 * cur + 2;maxv = arr[2 * cur + 2];}if (maxv > arr[cur]) {int tmp = arr[cur];arr[cur] = arr[index];arr[index] = tmp;}cur = index;}}
}

输出:

[3563, 634, 3423, 57, 537, 764, 76, 34, 6, 56, 57, 46, 536, 4, 6, 3, 33, 3, 1, 5, 5, 3, 34, 23, 4, 42, 65, 4]
[1, 3, 3, 3, 4, 4, 4, 5, 5, 6, 6, 23, 33, 34, 34, 42, 46, 56, 57, 57, 65, 76, 536, 537, 634, 764, 3423, 3563]

这篇关于堆排及时间复杂度分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

修改若依框架Token的过期时间问题

《修改若依框架Token的过期时间问题》本文介绍了如何修改若依框架中Token的过期时间,通过修改`application.yml`文件中的配置来实现,默认单位为分钟,希望此经验对大家有所帮助,也欢迎... 目录修改若依框架Token的过期时间修改Token的过期时间关闭Token的过期时js间总结修改若依

Go Mongox轻松实现MongoDB的时间字段自动填充

《GoMongox轻松实现MongoDB的时间字段自动填充》这篇文章主要为大家详细介绍了Go语言如何使用mongox库,在插入和更新数据时自动填充时间字段,从而提升开发效率并减少重复代码,需要的可以... 目录前言时间字段填充规则Mongox 的安装使用 Mongox 进行插入操作使用 Mongox 进行更

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

对postgresql日期和时间的比较

《对postgresql日期和时间的比较》文章介绍了在数据库中处理日期和时间类型时的一些注意事项,包括如何将字符串转换为日期或时间类型,以及在比较时自动转换的情况,作者建议在使用数据库时,根据具体情况... 目录PostgreSQL日期和时间比较DB里保存到时分秒,需要和年月日比较db里存储date或者ti

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g