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

一、先解答上次的思考题

对树: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. 依次取出堆顶,观察输出是否降序
相关推荐
通信小呆呆9 小时前
当算法有了“五感”:多模态数据融合如何向人体感官协同学习?
人工智能·学习·算法·机器学习·机器人
benben04410 小时前
强化学习之DQN算法族(基于gymnasium开发)
算法
小小工匠10 小时前
Redis - 事务机制:能实现 ACID 属性吗
数据结构·redis·性能优化·并发·持久化
玖玥拾11 小时前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
何以解忧,唯有..11 小时前
Go语言循环语句详解:for、range与循环控制
开发语言·算法·golang
想吃火锅100511 小时前
【leetcode】88.合并两个有序数组js
算法
生成论实验室12 小时前
机器人:一个自主运动的系统
人工智能·算法·语言模型·机器人·自动驾驶·agi·安全架构
Qres82112 小时前
算法复键——树状数组
数据结构·算法
H1785350909612 小时前
SolidWorks第四部分_直接实体建模特征9_替换面原理
线性代数·算法·机器学习·3d建模·solidworks