堆的实现(完全注释版本)

头文件

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
typedef int HPDataType;
typedef struct Heap
{
	HPDataType* a;
	int size;
	int capacity;
}Heap;
// 堆的初始化
void HeapInit(Heap* hp);
// 堆的销毁
void HeapDestory(Heap* hp);
// 堆的插入
void HeapPush(Heap* hp, HPDataType x);
// 堆的删除
void HeapPop(Heap* hp);
// 取堆顶的数据
HPDataType HeapTop(Heap* hp);
// 堆的数据个数
int HeapSize(Heap* hp);
// 堆的判空
bool HeapEmpty(Heap* hp);

函数实现

#define _CRT_SECURE_NO_WARNINGS 1
#include "t.h"
// 堆的初始化
void HeapInit(Heap* hp) {
    assert(hp);         // 确保堆指针不为空
    hp->a = NULL;       // 动态数组初始化为空
    hp->size = 0;       // 当前堆的大小初始化为0
    hp->capacity = 0;   // 当前堆的容量初始化为0
}
// 堆的销毁
void HeapDestory(Heap* hp) {
    assert(hp);         // 确保堆指针不为空
    free(hp->a);        // 释放动态数组的内存
    hp->a = NULL;       // 将指针置空,防止野指针
    hp->size = 0;       // 将大小重置为0
    hp->capacity = 0;   // 将容量重置为0
}
// 交换两个堆元素的值
void swap(HPDataType* px, HPDataType* py) {
	HPDataType newdata = *px;
	*px = *py;
	*py = newdata;
}
// 向上调整堆
void AdjustUp(HPDataType* a, int child) {
    int parent = (child - 1) / 2;  // 计算父节点的索引
    while (child > 0) {            // 如果当前节点不是根节点
        if (a[child] > a[parent]) { // 如果当前节点大于其父节点
            swap(&a[child], &a[parent]); // 交换
            child = parent;               // 更新当前节点为父节点
            parent = (parent - 1) / 2;    // 更新父节点索引
        } else {
            break;  // 如果满足堆性质,停止调整
        }
    }
}
// 向下调整堆
void AdjustDown(HPDataType* a, int n, int parent) {
    int child = parent * 2 + 1;  // 找到左孩子的索引
    while (child < n) {          // 当孩子节点在有效范围内
        // 找到较大的孩子节点
        if (child + 1 < n && a[child] < a[child + 1]) {
            ++child; // 更新为右孩子的索引
        }
        // 如果孩子大于父节点,交换并继续向下调整
        if (a[child] > a[parent]) {
            swap(&a[child], &a[parent]);
            parent = child;          // 更新父节点为孩子节点
            child = parent * 2 + 1;  // 更新左孩子索引
        } else {
            break; // 如果满足堆性质,停止调整
        }
    }
}
// 堆的插入
void HeapPush(Heap* hp, HPDataType x) {
    assert(hp);  // 确保堆指针不为空
    // 如果堆已满,则扩容
    if (hp->size == hp->capacity) {
        int newcapaity = hp->capacity == 0 ? 4 : hp->capacity * 2; // 扩容为当前容量的两倍
        HPDataType* tmp = (HPDataType*) realloc(hp->a, sizeof(HPDataType) * newcapaity);
        if (tmp == NULL) {  // 如果分配失败
            perror("realloc fail");
            return;
        }
        hp->a = tmp;           // 更新堆数组指针
        hp->capacity = newcapaity; // 更新堆容量
    }
    // 将新元素插入到堆尾,并向上调整
    hp->a[hp->size] = x;
    hp->size++;
    AdjustUp(hp->a, hp->size - 1);
}
// 堆的删除
void HeapPop(Heap* hp) {
    assert(hp);           // 确保堆指针不为空
    assert(hp->size > 0); // 确保堆不为空
    // 交换堆顶和堆尾元素
    swap(&hp->a[0], &hp->a[hp->size - 1]);
    hp->size--;           // 减少堆的大小
    AdjustDown(hp->a, hp->size, 0); // 调整堆顶元素以恢复堆性质
}
// 取堆顶的数据
HPDataType HeapTop(Heap* hp) {
	assert(hp);// 确保堆指针不为空
	return hp->a[0];
}
// 堆的数据个数
int HeapSize(Heap* hp){
	assert(hp);// 确保堆指针不为空
	return hp->size;
}
// 堆的判空
bool HeapEmpty(Heap* hp) {
	assert(hp);	// 确保堆指针不为空
	return hp->size == 0;
}

测试文件

#define _CRT_SECURE_NO_WARNINGS 1
#include"t.h"
int main(){
	int a[] = { 60,70,65,50,32,100 };
	Heap hp;
	HeapInit(&hp);
	for (int i = 0; i < sizeof(a) / sizeof(int); i++){
		HeapPush(&hp, a[i]);
	}
	while (!HeapEmpty(&hp)){
		printf("%d\n", HeapTop(&hp));
		HeapPop(&hp);
	}
	HeapDestory(&hp);
	return 0;
}

实现效果

什么嘛,还是挺有意思的

相关推荐
USER_A0017 分钟前
【C语言】第五期——函数
c语言
计算机小白一个5 小时前
蓝桥杯 Java B 组之设计 LRU 缓存
java·算法·蓝桥杯
万事可爱^6 小时前
HDBSCAN:密度自适应的层次聚类算法解析与实践
算法·机器学习·数据挖掘·聚类·hdbscan
李白同学6 小时前
【C语言】结构体内存对齐问题
c语言·开发语言
楼台的春风7 小时前
【MCU驱动开发概述】
c语言·驱动开发·单片机·嵌入式硬件·mcu·自动驾驶·嵌入式
大数据追光猿8 小时前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!8 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉8 小时前
力扣-回溯-46 全排列
数据结构·算法·leetcode
南宫生8 小时前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
柠石榴8 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯