排序算法之堆排序详细解读(附带Java代码解读)

2024-08-28 14:04

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

堆排序(Heap Sort)是一种基于比较的排序算法,它利用堆数据结构来排序元素。堆是一种特殊的完全二叉树,堆排序的基本思想是将数组构建成一个最大堆(或最小堆),然后通过交换根节点和堆的最后一个元素,将最大(或最小)元素移到数组的末尾。接着,调整堆,使其重新满足堆的性质,然后重复这一过程直到排序完成。

算法思想

  1. 构建最大堆:将无序数组构建成一个最大堆。最大堆的特性是每个节点的值都大于或等于其子节点的值。
  2. 提取最大元素:将堆顶(最大元素)与堆的最后一个元素交换,然后将堆的有效大小减小 1。对交换后的堆进行调整,以保持最大堆的性质。
  3. 重复步骤 2:直到堆的有效大小为 1,整个数组已经排序完成。

过程示例

假设有一个待排序的数组:[4, 10, 3, 5, 1]

步骤 1: 构建最大堆

将数组构建成最大堆:

  • 初始数组:[4, 10, 3, 5, 1]
  • 从最后一个非叶子节点开始向上调整:
    • 调整节点 10:[10, 5, 3, 4, 1]
    • 调整节点 4:[10, 5, 4, 3, 1]

步骤 2: 提取最大元素

将堆顶(最大元素 10)与堆的最后一个元素交换,然后调整堆:

  • 交换:[1, 5, 4, 3, 10]
  • 调整堆:[5, 3, 4, 1, 10]
  • 交换堆顶和最后一个元素后,堆变成:[5, 3, 4, 1],最大元素 10 已经放在正确位置

重复上述步骤:

  • 交换堆顶和最后一个元素:[1, 3, 4, 5, 10]
  • 调整堆:[4, 3, 1, 5, 10]
  • 交换堆顶和最后一个元素:[1, 3, 4, 5, 10]
  • 调整堆:[3, 1, 4, 5, 10]
  • 交换堆顶和最后一个元素:[1, 3, 4, 5, 10]
  • 调整堆:[1, 3, 4, 5, 10],最终数组为:[1, 3, 4, 5, 10]

算法复杂度

  • 时间复杂度:

    • 最坏情况: O(n log n)
    • 平均情况: O(n log n)
    • 最佳情况: O(n log n)
  • 空间复杂度: O(1) 堆排序是原地排序算法,不需要额外的存储空间。

优点

  1. 原地排序:只需要常数级别的额外空间。
  2. 时间复杂度优良:最坏情况时间复杂度为 O(n log n),适合大数据量的排序。

缺点

  1. 不稳定排序:相等元素的相对顺序可能会改变。
  2. 实现较复杂:相较于其他排序算法(如插入排序),堆排序的实现较复杂。

Java代码解读

public class HeapSort {// 主方法:执行堆排序public static void heapSort(int[] arr) {int n = arr.length;// 构建最大堆for (int i = n / 2 - 1; i >= 0; i--) {heapify(arr, n, i);}// 提取元素并调整堆for (int i = n - 1; i > 0; i--) {// 将当前堆顶(最大值)交换到数组末尾int temp = arr[0];arr[0] = arr[i];arr[i] = temp;// 调整堆heapify(arr, i, 0);}}// 调整堆的函数private static void heapify(int[] arr, int n, int i) {int largest = i; // 初始化最大元素为根节点int left = 2 * i + 1; // 左子节点索引int right = 2 * i + 2; // 右子节点索引// 如果左子节点大于根节点if (left < n && arr[left] > arr[largest]) {largest = left;}// 如果右子节点大于根节点if (right < n && arr[right] > arr[largest]) {largest = right;}// 如果最大元素不是根节点if (largest != i) {int swap = arr[i];arr[i] = arr[largest];arr[largest] = swap;// 递归地调整被替换子节点的堆heapify(arr, n, largest);}}public static void main(String[] args) {int[] arr = {4, 10, 3, 5, 1};System.out.println("排序前的数组:");for (int num : arr) {System.out.print(num + " ");}System.out.println();heapSort(arr);System.out.println("排序后的数组:");for (int num : arr) {System.out.print(num + " ");}}
}

代码说明

  1. heapSort方法:

    • heapSort 方法首先构建最大堆,然后逐步将最大元素移到数组末尾,并调整堆。
  2. heapify方法:

    • heapify 方法调整堆以保持最大堆的性质。
    • 它检查节点的左子节点和右子节点,将最大元素移动到根节点,并递归地调整被替换子节点的堆。
  3. main方法:

    • 创建一个待排序的数组 arr
    • 调用 heapSort 方法对数组进行排序。
    • 输出排序前和排序后的数组。

 

这篇关于排序算法之堆排序详细解读(附带Java代码解读)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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

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

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

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

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

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

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

Java利用docx4j+Freemarker生成word文档

《Java利用docx4j+Freemarker生成word文档》这篇文章主要为大家详细介绍了Java如何利用docx4j+Freemarker生成word文档,文中的示例代码讲解详细,感兴趣的小伙伴... 目录技术方案maven依赖创建模板文件实现代码技术方案Java 1.8 + docx4j + Fr

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

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:变长字符数

基于SpringBoot+Mybatis实现Mysql分表

《基于SpringBoot+Mybatis实现Mysql分表》这篇文章主要为大家详细介绍了基于SpringBoot+Mybatis实现Mysql分表的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录基本思路定义注解创建ThreadLocal创建拦截器业务处理基本思路1.根据创建时间字段按年进