数据结构之堆

堆的定义与基本概念

堆是一种特殊的完全二叉树,也叫优先级队列,满足以下性质之一:

  • 最大堆:任意节点的值大于或等于其子节点的值(根节点为最大值)。
  • 最小堆:任意节点的值小于或等于其子节点的值(根节点为最小值)。

堆的存储方式

堆通常通过数组实现,利用完全二叉树的特性进行索引映射:

  • 对于节点 i(从 0 开始计数):
    • 父节点索引:(i - 1) // 2
    • 左子节点索引:2i + 1
    • 右子节点索引:2i + 2

堆的操作

插入元素(上浮)向上调整多用于插入

  1. 将新元素添加到数组末尾。
  2. 比较新元素与其父节点的值,若违反堆性质则交换。
  3. 重复上浮过程直至满足堆性质。

删除堆顶元素(下沉)向下调整多用于构建堆和删除元素

  1. 交换堆顶与数组末尾元素,并删除末尾元素。
  2. 从堆顶开始,比较当前节点与子节点的值,若违反堆性质则与较大(最大堆)或较小(最小堆)的子节点交换。
  3. 重复下沉过程直至满足堆性质。

堆的构建

时间复杂度O(n)(自底向上调整)。

  1. 从最后一个非叶子节点开始(索引 n//2 - 1)。
  2. 对每个节点执行下沉操作,确保子树满足堆性质。

代码示例(Java)

java 复制代码
import java.util.ArrayList;
import java.util.Arrays;

/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: zhany
 * Date: 2025-10-24
 * Time: 19:48
 */
public class Piorory {
    int[] elem;
    int usedSize;
    public Piorory(){
        this.elem = new int[10];
    }
    public void initElem(int[] array) {
        for (int i = 0; i < array.length; i++) {
            this.elem[i] = array[i];
            this.usedSize++;
        }
    }
    public void creatHeap(){
        for (int parent = (this.usedSize-2)/2; parent >=0; parent--) {
            siftDown(parent,this.usedSize);
        }
    }
//向下调整主要是用于构建堆和删除顶部元素
    private void siftDown(int parent, int usedSize) {
        int child = 2*parent +1;
        while(child<usedSize){
            if(child+1<usedSize && elem[child] <elem[child+1]){
                child ++;
            }
            if(elem[child] >elem[parent]){
                swap(elem,child,parent);
                parent = child;
                child = 2*parent+1;
            } else{
                break;
            }
        }
    }

    private void swap(int[]elem,int child, int parent) {
        int tmp = elem[child];
        elem[child] = elem[parent];
        elem[parent] = tmp;
    }
    //向上主要是用于插入元素
    public void shifUp(int child){
        int parent = (child-1)/2;
        while(parent >= 0){
            if (elem[child] > elem[parent]) {
                swap(elem,child,parent);
                child = parent;
                parent = (child -1)/2;
            } else{
                break;
            }
        }
    }

    public void offer(int val){
        if(isFull()){
        elem = Arrays.copyOf(elem,2*elem.length);
    }
        elem[usedSize] = val;
        shifUp(usedSize);
        usedSize++;
    }
    public  boolean isFull(){
        return  usedSize == elem.length;
    }
    public boolean isEmpty(){
        return usedSize == 0;
    }
    public int peek(){
        if (isEmpty()) {
            return -1;
        }
            return elem[0];
    }
    public void printHeap() {
        for (int i = 0; i < usedSize; i++) {
            System.out.print(elem[i] + " ");
        }
        System.out.println();
    }
}
相关推荐
凌晨一点的秃头猪3 小时前
Python 常见 bug 总结和异常处理
开发语言·python·bug
mortimer3 小时前
用PySide6 构建一个响应式视频剪辑工具:多线程与信号机制实战
python·ffmpeg·pyqt
新子y3 小时前
【小白笔记】input() 和 print() 这两个函数
笔记·python
SleepyWhite0013 小时前
代码随想录Day61|Floyd 算法精讲、A * 算法精讲
算法·floyd算法·astar算法
Miraitowa_cheems3 小时前
LeetCode算法日记 - Day 84: 乘积为正数的最长子数组长度
数据结构·算法·leetcode·贪心算法·线性回归·深度优先·动态规划
雾岛听蓝3 小时前
C语言:使用顺序表实现通讯录
c语言·数据结构·经验分享·笔记·visualstudio
文火冰糖的硅基工坊4 小时前
[人工智能-大模型-72]:模型层技术 - 模型训练六大步:①数据预处理 - 基本功能与对应的基本组成函数
开发语言·人工智能·python
不是老弟4 小时前
rwqsd
数据结构·c++·算法