java堆排序概念原理介绍

作者:laozhang 时间:2021-08-30 12:31:58 

堆排序介绍:

堆排序可以分为两个阶段。在堆的构造阶段,我们将原始数组重新组织安排进一个堆中;然后在下沉排序阶段,我们从堆中按顺序取出所有元素并得到排序结果。

1.堆的构造,一个有效的方法是从右到左使用sink()下沉函数构造子堆。数组的每个位置都有一个子堆的根节点,sink()对于这些子堆也适用,如果一个节点的两个子节点都已经是堆了,那么在该节点上调用sink()方法可以把他们合并成一个堆。我们可以跳过大小为1的子堆,因为大小为1的不需要sink()也就是下沉操作,有关下沉和上浮操作可以参考我写的优先队列那篇。

2.堆的排序,我们通过第一步操作构造了一个堆,在这个堆中,根节点永远是最大值的节点,所以我们把根节点的值与数组最后的值进行交换,在使用sink()下沉来维护堆的结构即可。

具体代码实现:


public class PQSort{
 public static void main(String[] args){
   int[] a = {9,1,7,5,3,9,12,56,21,45};
   sort(a);
   for(int i=0;i<a.length;i++) {
     System.out.print(a[i]+" ");
   }  
 }
 //排序方法
 public static void sort(int[] a){
     int N = a.length-1;
     //通过下沉操作构造堆,因为下标从0开始,所以子节点为2*k+1和2*k+2;
     for(int k = (N-2)/2;k>=0;k--){
       sink(a,k,N);
     }
     //通过不断把堆中最大值放到数组的后面来排序
     while(N>0){
       exch(a,0,N--);
       sink(a,0,N);
     }
 }
 //下沉函数
 private static void sink(int[] a, int i, int n){
   while(2*i+1<=n){
     int j = 2*i+1;
     if(j<n&&a[j]<a[j+1]) j++;
     if(a[i]>a[j]) break;
     exch(a,i,j);
     i=j;
   }
 }
 //交换函数
 private static void exch(int[] a, int i, int j){
   int temp = a[i];
   a[i] = a[j];
   a[j] = temp;
 }
}

运行结果:

java堆排序概念原理介绍

标签:java,堆排序
0
投稿

猜你喜欢

  • Java 可视化垃圾回收_动力节点Java学院整理

    2023-02-19 07:03:55
  • Java关键字、标识符、常量、变量语法详解

    2023-11-11 20:49:48
  • C++常见错误中英文对照表

    2023-06-05 04:52:13
  • 微信公众号开发之回复图文消息java代码

    2022-11-24 00:56:19
  • C#正则表达式Regex类的用法

    2022-06-07 13:44:58
  • C# 执行CMD命令并接收返回结果的操作方式

    2023-09-03 10:36:56
  • Springboot整合企业微信机器人助手推送消息的实现

    2023-08-18 08:37:37
  • Android拍照或从图库选择图片并裁剪

    2023-07-01 08:21:45
  • Unity UI实现拖拽旋转

    2023-04-10 06:44:59
  • Android Volley图片加载功能详解

    2023-11-19 08:01:41
  • Flutter实现固定header底部滑动页效果示例

    2022-06-15 06:31:05
  • c# 利用易福门振动模块VSE002采集振动数据的方法

    2022-10-13 00:12:25
  • 图文详解OkHttp的超时时间

    2022-05-14 13:50:23
  • java使用EasyExcel导入导出excel

    2022-10-31 03:48:30
  • SpringBoot 监控管理模块actuator没有权限的问题解决方法

    2022-01-26 21:50:11
  • 使用SpringBoot获取所有接口的路由

    2021-08-29 22:28:34
  • Android自定义View实现垂直时间轴布局

    2022-12-12 15:21:16
  • c#判断磁盘驱动器类型的两种方法介绍

    2023-12-18 10:04:53
  • kafka消费不到数据的排查过程

    2023-11-14 04:35:30
  • Java实现中文算数验证码的实现示例(算数运算+-*/)

    2023-10-23 03:08:09
  • asp之家 软件编程 m.aspxhome.com