java实现堆排序(详细解释代码和逻辑)

简介: java实现堆排序(详细解释代码和逻辑)

堆排序(Heap Sort)是一种基于堆数据结构的比较排序算法。堆是一个近似完全二叉树的结构,满足堆属性:即对于最大堆,任何一个节点的值都大于或等于其子节点的值;对于最小堆,任何一个节点的值都小于或等于其子节点的值。


下面的代码例子展示了堆排序的实现,包括构建最大堆、堆排序主函数和辅助函数:

// Java 实现的堆排序
public class HeapSort {
 
    // 主函数,用于对输入数组进行排序
    public void sort(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);
        }
    }
 
    // 辅助函数,用于维护堆属性
    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[] = {12, 11, 13, 5, 6, 7};
        int n = arr.length;
 
        HeapSort ob = new HeapSort();
        ob.sort(arr);
 
        System.out.println("排序后的数组是:");
        printArray(arr);
    }
 
    // 用于打印数组
    static void printArray(int arr[]) {
        int n = arr.length;
        for (int i = 0; i < n; ++i) {
            System.out.print(arr[i] + " ");
        }
        System.out.println();
    }
}

代码解析:

构建最大堆: 在sort方法中,我们首先从第一个非叶子节点开始,从右到左从下到上地对每个节点调用heapify方法,这样就能将输入数组构建成一个最大堆。


堆排序: 通过从数组的最后一个元素开始,逐步将根节点(最大值)与当前元素交换,然后对缩小后的堆进行堆化,从而使数组从小到大排序。


堆化(heapify): heapify方法用于维护堆属性。它通过比较父节点与其子节点的大小,如果子节点大于父节点,则交换它们的位置,并递归地对受影响的子树进行堆化。


打印数组: printArray方法用于输出排序后的数组。

第二个代码例子

这里是另一个堆排序的实现例子,这次使用最小堆来进行排序:

// Java 实现的最小堆排序
public class MinHeapSort {
 
    // 主函数,用于对输入数组进行排序
    public void sort(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);
        }
    }
 
    // 辅助函数,用于维护最小堆属性
    void heapify(int arr[], int n, int i) {
        int smallest = i; // 初始化为根节点
        int left = 2 * i + 1; // 左子节点
        int right = 2 * i + 2; // 右子节点
 
        // 如果左子节点比根节点小
        if (left < n && arr[left] < arr[smallest]) {
            smallest = left;
        }
 
        // 如果右子节点比当前最小值小
        if (right < n && arr[right] < arr[smallest]) {
            smallest = right;
        }
 
        // 如果最小值不是根节点,则交换它们
        if (smallest != i) {
            int swap = arr[i];
            arr[i] = arr[smallest];
            arr[smallest] = swap;
 
            // 递归地对受影响的子树进行堆化
            heapify(arr, n, smallest);
        }
    }
 
    // 测试函数
    public static void main(String args[]) {
        int arr[] = {12, 11, 13, 5, 6, 7};
        int n = arr.length;
 
        MinHeapSort ob = new MinHeapSort();
        ob.sort(arr);
 
        System.out.println("排序后的数组是:");
        printArray(arr);
    }
 
    // 用于打印数组
    static void printArray(int arr[]) {
        int n = arr.length;
        for (int i = 0; i < n; ++i) {
            System.out.print(arr[i] + " ");
        }
        System.out.println();
    }
}

代码解析:

构建最小堆: 在sort方法中,我们首先从第一个非叶子节点开始,从右到左从下到上地对每个节点调用heapify方法,这样就能将输入数组构建成一个最小堆。


堆排序: 通过从数组的最后一个元素开始,逐步将根节点(最小值)与当前元素交换,然后对缩小后的堆进行堆化,从而使数组从大到小排序。


堆化(heapify): heapify方法用于维护最小堆属性。它通过比较父节点与其子节点的大小,如果子节点小于父节点,则交换它们的位置,并递归地对受影响的子树进行堆化。


通过这两个例子,我们可以看到堆排序的主要步骤和其实现方式。堆排序的时间复杂度为O(n log n),它是一种不稳定的排序算法,但它的空间复杂度为O(1),因为它是就地排序(in-place sorting)。


相关文章
|
2月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
79 1
|
2月前
|
Java
在Java中实现接口的具体代码示例
可以根据具体的需求,创建更多的类来实现这个接口,以满足不同形状的计算需求。希望这个示例对你理解在 Java 中如何实现接口有所帮助。
92 38
|
13天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
35 3
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
56 24
|
20天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
56 2
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
82 5
|
1月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
57 5
|
2月前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
|
2月前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
50 5
Java反射机制:解锁代码的无限可能
|
2月前
|
Java API Maven
商汤人像如何对接?Java代码如何写?
商汤人像如何对接?Java代码如何写?
51 5