堆排序:高效构建大顶堆实战

一、先解答上次的思考题

对树:10,7,15,3,9,12,18

  • 删除 15 后中序:3 7 9 10 12 18
  • 再删除 7 后中序:3 9 10 12 18

二、今天学习目标

  1. 什么是堆、大顶堆、小顶堆
  2. 堆的结构:用数组存完全二叉树
  3. 核心操作:上浮、下沉
  4. 堆实现优先队列(完整代码)

三、什么是堆(Heap)?

堆是一棵满足堆序性质的完全二叉树 ,通常用数组存储。

两种常用堆:

  • 大顶堆:父节点 ≥ 左右孩子,堆顶是最大值
  • 小顶堆:父节点 ≤ 左右孩子,堆顶是最小值

特点:

  • 结构是完全二叉树
  • 可以用数组高效存储
  • 取最值极快:O (1)
  • 插入 / 删除:O (log n)

四、数组存储完全二叉树(下标规律)

对于下标 i(从 0 开始):

  • 左孩子:2*i + 1
  • 右孩子:2*i + 2
  • 父节点:(i - 1) / 2

五、核心操作:下沉(向下调整)

大顶堆为例:如果某个节点比孩子小,就和较大的孩子交换,一直往下沉,直到满足堆性质。

cpp 复制代码
// 大顶堆下沉调整
void siftDown(int arr[], int n, int i) {
    while (1) {
        int maxPos = i;
        int left = 2 * i + 1;
        int right = 2 * i + 2;

        if (left < n && arr[left] > arr[maxPos]) maxPos = left;
        if (right < n && arr[right] > arr[maxPos]) maxPos = right;

        if (maxPos == i) break;

        // 交换
        int temp = arr[i];
        arr[i] = arr[maxPos];
        arr[maxPos] = temp;

        i = maxPos;
    }
}

六、建堆 + 堆实现优先队列(取堆顶)

完整可运行代码

cpp 复制代码
#include <stdio.h>

// 下沉调整(大顶堆)
void siftDown(int arr[], int n, int i) {
    while (1) {
        int maxPos = i;
        int left = 2 * i + 1;
        int right = 2 * i + 2;

        if (left < n && arr[left] > arr[maxPos])
            maxPos = left;
        if (right < n && arr[right] > arr[maxPos])
            maxPos = right;

        if (maxPos == i)
            break;

        int temp = arr[i];
        arr[i] = arr[maxPos];
        arr[maxPos] = temp;

        i = maxPos;
    }
}

// 建堆
void buildHeap(int arr[], int n) {
    // 从最后一个非叶子节点往前建堆
    for (int i = n / 2 - 1; i >= 0; i--) {
        siftDown(arr, n, i);
    }
}

// 删除堆顶(取最大值)
int deleteMax(int arr[], int *n) {
    int maxVal = arr[0];
    // 最后一个元素放到堆顶
    arr[0] = arr[*n - 1];
    (*n)--;
    siftDown(arr, *n, 0);
    return maxVal;
}

// ==================== 测试 ====================
int main() {
    int arr[] = {3, 1, 7, 9, 4, 2, 8};
    int n = sizeof(arr) / sizeof(arr[0]);

    buildHeap(arr, n);

    printf("依次取出堆顶(大顶堆):");
    while (n > 0) {
        printf("%d ", deleteMax(arr, &n));
    }

    return 0;
}

七、运行结果

cpp 复制代码
依次取出堆顶(大顶堆):9 8 7 4 3 2 1

八、堆的典型用途

  • 优先队列:任务调度、定时器
  • 堆排序:O (n log n) 稳定高效
  • TopK 问题:找最大 / 最小的 k 个数

九、今日小练习

  1. 把数组 [5, 2, 8, 1, 3] 建成大顶堆
  2. 依次取出堆顶,观察输出是否降序
相关推荐
深邃-5 分钟前
【数据结构与算法】-二叉树(2):实现顺序结构二叉树(堆的实现),向上调整算法,向下调整算法,堆排序,TOP-K问题
数据结构·算法·二叉树·排序算法·堆排序··top-k
We་ct3 小时前
LeetCode 5. 最长回文子串:DP + 中心扩展
前端·javascript·算法·leetcode·typescript
王老师青少年编程7 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
叼烟扛炮8 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
天疆说8 小时前
【哈密顿力学】深入解读航天器交会最优控制中的Hamilton函数
人工智能·算法·机器学习
wuweijianlove8 小时前
关于算法设计中的代价函数优化与约束求解的技术7
算法
leoufung9 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了9 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL9 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
谭欣辰9 小时前
C++ 排列组合完整指南
开发语言·c++·算法