堆排序-java

堆排序基本介绍

堆排序是利用堆这种数据结构而设计的一种排序算法,堆排序是一种选择排序,它的最坏,最好,平均时间复杂度均为O(nlogn),它也是不稳定排序。
堆是具有以下性质的完全二叉树:
大顶堆:
每个结点的值都大于或等于其左右孩子结点的值,注意 : 没有要求结点的左孩子的值和右孩子的值的大小关系。
小顶堆:
每个结点的值都小于或等于其左右孩子结点的值

大顶堆举例说明
在这里插入图片描述
在这里插入图片描述

堆排序基本思想

  1. 将待排序序列构造成一个大顶堆
  2. 此时,整个序列的最大值就是堆顶的根节点。
  3. 将其与末尾元素进行交换,此时末尾就为最大值。
  4. 然后将剩余n-1个元素重新构造成一个堆,这样会得到n个元素的次小值。如此反复执行,便能得到一个有序序列了。
package sortalgorithm;

import java.util.Arrays;

/**
 * @author: xingt   mym_74@163.com
 * @date: 2019/11/1, 16:56
 * @version: 1.0
 * 平均时间复杂度:O(nlogn)
 * 空间复杂度 O(1)
 * 稳定性: 不稳定
 */
public class HeapSort {
    public static void main(String[] args) {
        int[] arr1 = {1, 4, 6, 8, 2, 5, 3, 7, 9};
        System.out.println("排序前的数组为:" + Arrays.toString(arr1));
        //mergeSort(arr1,0,arr1.length-1);
        heapSort(arr1);

        System.out.println("排序后的数组为:" + Arrays.toString(arr1));
    }


    public static void heapSort(int[] arr){
        //构建大顶堆  arr.length/2-1 就是代表最后一个非叶子节点
        for(int i= arr.length/2 -1; i>=0;i--){
            adjustHeap(arr,i,arr.length);
        }
        //将堆顶元素与末尾元素进行交换,将最大元素"沉"到素组末尾
        //重新调整结构,是其满足堆定义,然后继续交换堆顶元素与当前末尾元素,反复执行
        // j=arr.length-1 是因为eg:总共5个数,只需要调整4个数就可以了
        for(int j = arr.length-1;j>0;j--){
            //索引为0的数是大顶堆最大的值
            swap(arr,0,j);
            //重新对堆进行调整
            //以0作为初始调整的索引,比较方便, j是每次循环都要排除掉最后一个已经排好序的值
            adjustHeap(arr,0,j);
        }

    }


    /**
     * 调整数组为大顶堆
     * 举例: int arr[] = {4,6,8,5,9}; => i=1 (也就是6) => adjustHeap => 得到{4,9,8,5,6}
     * 如果再次调用adjustHeap 传入的是 i=0 => 得到{4,9,8,5,6} => {9,6,8,5,4}
     * @param arr 传入的数组
     * @param i 需要调整的数的索引
     * @param length 数组的长度
     */
    public static void adjustHeap(int[] arr, int i , int length){
        //取出当前索引为index的元素
        int temp = arr[i];
        // k= i*2+1 当前节点的子左节点
        //从左子节点开始遍历,下一次调整也是再下一个左子节点
        for(int k=i*2+1; k<length; k = k*2+1){
            //如果左子节点值小于右子节点值,k+1 指向右节点
            if(k+1<length && arr[k]<arr[k+1]){
                k++; //指向右子节点 k+1
            }
            //如果子节点大于父节点,将子节点值赋给父节点(不交换)
            if(arr[k]>temp){
                arr[i] = arr[k];
                // !!!把缱当前索引index指向k, 继续循环比较
                i = k;
                //与后面的arr[i] = temp; 合在一起的话就相当于交换数值,不过代码中需要for循环找到那个最大的值
            }else{
                break;
            }

        }//for循环结束后,我们已经把以index为父节点的的树的最大值,调整在了顶部(局部)
        //将temp值放到最终的位置
        arr[i] = temp;

    }

    public static void swap(int[] arr,int a, int b){
        int temp = arr[a];
        arr[a] = arr[b];
        arr[b] = temp;
    }


}

堆排序是一种常见的排序算法,具有稳定性高、效率高等优点。下面介绍一下使用Java实现的小根堆排序。 首先,我们需要定义一个小根堆类,用于存储待排序的数据。该类需要包含以下几个方法: 1. `Heap()`:构造函数,用于初始化小根堆。 2. `insert(int val)`:插入操作,将一个新的元素插入到小根堆中。 3. `deleteMin()`:删除操作,删除小根堆中的最小元素,并返回该元素的值。 4. `size()`:获取小根堆中元素的个数。 5. `isEmpty()`:判断小根堆是否为空。 接下来,我们就可以使用小根堆对待排序的数据进行排序了。具体的步骤如下: 1. 将待排序的数据存入小根堆中。 2. 依次从小根堆中删除最小元素,并将其存入数组中。 3. 最后,将数组反转,即可得到排序后的结果。 下面是具体的Java代码实现: ```java public class HeapSort { public static void heapSort(int[] arr) { Heap heap = new Heap(arr.length); for (int i = 0; i < arr.length; i++) { heap.insert(arr[i]); } for (int i = 0; i < arr.length; i++) { arr[i] = heap.deleteMin(); } // 反转数组 reverse(arr); } // 反转数组 private static void reverse(int[] arr) { int left = 0; int right = arr.length - 1; while (left < right) { int temp = arr[left]; arr[left] = arr[right]; arr[right] = temp; left++; right--; } } // 小根堆类 static class Heap { private int[] heap; private int size; public Heap(int capacity) { heap = new int[capacity + 1]; size = 0; } public void insert(int val) { if (size == heap.length - 1) { throw new RuntimeException("Heap is full"); } int i = ++size; while (i != 1 && val < heap[i / 2]) { heap[i] = heap[i / 2]; i /= 2; } heap[i] = val; } public int deleteMin() { if (isEmpty()) { throw new RuntimeException("Heap is empty"); } int min = heap[1]; int last = heap[size--]; int i = 1; int child = 2; while (child <= size) { if (child < size && heap[child] > heap[child + 1]) { child++; } if (last > heap[child]) { heap[i] = heap[child]; i = child; child *= 2; } else { break; } } heap[i] = last; return min; } public int size() { return size; } public boolean isEmpty() { return size == 0; } } } ``` 使用该算法对数组进行排序: ```java int[] arr = {3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5}; HeapSort.heapSort(arr); System.out.println(Arrays.toString(arr)); // 输出 [1, 1, 2, 3, 3, 4, 5, 5, 5, 6, 9] ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值